diff.cpp 2.3 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374
  1. /*
  2. * Copyright (c) 2021, Mustafa Quraish <mustafa@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <LibCore/ArgsParser.h>
  7. #include <LibCore/File.h>
  8. #include <LibDiff/Generator.h>
  9. #include <unistd.h>
  10. int main(int argc, char** argv)
  11. {
  12. if (pledge("stdio rpath", nullptr) < 0) {
  13. perror("pledge");
  14. return 1;
  15. }
  16. Core::ArgsParser parser;
  17. char const* filename1;
  18. char const* filename2;
  19. parser.add_positional_argument(filename1, "First file to compare", "file1", Core::ArgsParser::Required::Yes);
  20. parser.add_positional_argument(filename2, "Second file to compare", "file2", Core::ArgsParser::Required::Yes);
  21. parser.parse(argc, argv, Core::ArgsParser::FailureBehavior::PrintUsageAndExit);
  22. auto file1 = Core::File::construct(filename1);
  23. if (!file1->open(Core::OpenMode::ReadOnly)) {
  24. warnln("Error: Cannot open {}: {}", filename1, file1->error_string());
  25. return 1;
  26. }
  27. auto file2 = Core::File::construct(filename2);
  28. if (!file2->open(Core::OpenMode::ReadOnly)) {
  29. warnln("Error: Cannot open {}: {}", filename2, file2->error_string());
  30. return 1;
  31. }
  32. auto hunks = Diff::from_text(file1->read_all(), file2->read_all());
  33. for (const auto& hunk : hunks) {
  34. auto original_start = hunk.original_start_line;
  35. auto target_start = hunk.target_start_line;
  36. auto num_added = hunk.added_lines.size();
  37. auto num_removed = hunk.removed_lines.size();
  38. StringBuilder sb;
  39. // Source line(s)
  40. sb.appendff("{}", original_start);
  41. if (num_removed > 1)
  42. sb.appendff(",{}", original_start + num_removed - 1);
  43. // Action
  44. if (num_added > 0 && num_removed > 0)
  45. sb.append("c");
  46. else if (num_added > 0)
  47. sb.append("a");
  48. else
  49. sb.append("d");
  50. // Target line(s)
  51. sb.appendff("{}", target_start);
  52. if (num_added > 1)
  53. sb.appendff(",{}", target_start + num_added - 1);
  54. outln("Hunk: {}", sb.build());
  55. for (const auto& line : hunk.removed_lines)
  56. outln("\033[31;1m< {}\033[0m", line);
  57. if (num_added > 0 && num_removed > 0)
  58. outln("---");
  59. for (const auto& line : hunk.added_lines)
  60. outln("\033[32;1m> {}\033[0m", line);
  61. }
  62. return hunks.is_empty() ? 0 : 1;
  63. }