/* * Copyright (c) 2022, Sam Atkins * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include static ErrorOr> open_file_or_stdin(DeprecatedString const& filename) { OwnPtr file; if (filename == "-") { file = TRY(Core::Stream::File::adopt_fd(STDIN_FILENO, Core::Stream::OpenMode::Read)); } else { file = TRY(Core::Stream::File::open(filename, Core::Stream::OpenMode::Read)); } return TRY(Core::Stream::BufferedFile::create(file.release_nonnull())); } ErrorOr serenity_main(Main::Arguments arguments) { Main::set_return_code_for_errors(2); TRY(Core::System::pledge("stdio rpath")); Core::ArgsParser parser; DeprecatedString filename1; DeprecatedString filename2; bool verbose = false; bool silent = false; parser.set_general_help("Compare two files, and report the first byte that does not match. Returns 0 if files are identical, or 1 if they differ."); parser.add_positional_argument(filename1, "First file to compare", "file1", Core::ArgsParser::Required::Yes); parser.add_positional_argument(filename2, "Second file to compare", "file2", Core::ArgsParser::Required::Yes); parser.add_option(verbose, "Output every byte mismatch, not just the first", "verbose", 'l'); parser.add_option(silent, "Disable all output", "silent", 's'); parser.parse(arguments); // When opening STDIN as both files, the results are undefined. // Let's just report that it matches. if (filename1 == "-" && filename2 == "-") return 0; auto file1 = TRY(open_file_or_stdin(filename1)); auto file2 = TRY(open_file_or_stdin(filename2)); TRY(Core::System::unveil(nullptr, nullptr)); int line_number = 1; int byte_number = 1; Array buffer1; Array buffer2; bool files_match = true; auto report_mismatch = [&]() { files_match = false; if (silent) return; if (verbose) outln("{} {:o} {:o}", byte_number, buffer1[0], buffer2[0]); else outln("{} {} differ: char {}, line {}", filename1, filename2, byte_number, line_number); }; auto report_eof = [&](auto& shorter_file_name) { files_match = false; if (silent) return; auto additional_info = verbose ? DeprecatedString::formatted(" after byte {}", byte_number) : DeprecatedString::formatted(" after byte {}, line {}", byte_number, line_number); warnln("cmp: EOF on {}{}", shorter_file_name, additional_info); }; while (true) { TRY(file1->read(buffer1)); TRY(file2->read(buffer2)); if (file1->is_eof() && file2->is_eof()) break; if (file1->is_eof() || file2->is_eof()) { report_eof(file1->is_eof() ? filename1 : filename2); break; } if (buffer1[0] != buffer2[0]) { report_mismatch(); if (!verbose) break; } if (buffer1[0] == '\n') ++line_number; ++byte_number; } return files_match ? 0 : 1; }