diff.cpp 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  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. bool color_output = isatty(STDOUT_FILENO);
  55. outln("Hunk: {}", sb.build());
  56. for (const auto& line : hunk.removed_lines) {
  57. if (color_output)
  58. outln("\033[31;1m< {}\033[0m", line);
  59. else
  60. outln("< {}", line);
  61. }
  62. if (num_added > 0 && num_removed > 0)
  63. outln("---");
  64. for (const auto& line : hunk.added_lines) {
  65. if (color_output)
  66. outln("\033[32;1m> {}\033[0m", line);
  67. else
  68. outln("> {}", line);
  69. }
  70. }
  71. return hunks.is_empty() ? 0 : 1;
  72. }