vcpkg/toolsrc/src/commands_portsdiff.cpp

161 lines
7.5 KiB
C++
Raw Normal View History

2017-01-28 04:49:09 +08:00
#include "pch.h"
2016-11-07 12:12:54 +08:00
#include "vcpkg_Commands.h"
#include "vcpkg_System.h"
#include "vcpkg_Maps.h"
#include "SourceParagraph.h"
2017-02-28 07:52:57 +08:00
#include "Paragraphs.h"
#include "SortedVector.h"
2016-11-07 12:12:54 +08:00
namespace vcpkg::Commands::PortsDiff
2016-11-07 12:12:54 +08:00
{
2017-03-30 09:14:48 +08:00
struct updated_port
{
static bool compare_by_name(const updated_port& left, const updated_port& right)
{
return left.port < right.port;
}
std::string port;
version_diff_t version_diff;
};
2017-03-30 08:33:08 +08:00
template <class T>
struct set_element_presence
2016-11-07 12:12:54 +08:00
{
2017-03-30 08:33:08 +08:00
static set_element_presence create(std::vector<T> left, std::vector<T> right)
2016-11-07 12:12:54 +08:00
{
2017-03-30 08:33:08 +08:00
// TODO: This can be done with one pass instead of three passes
set_element_presence output;
std::set_difference(left.cbegin(), left.cend(), right.cbegin(), right.cend(), std::back_inserter(output.only_left));
std::set_intersection(left.cbegin(), left.cend(), right.cbegin(), right.cend(), std::back_inserter(output.both));
std::set_difference(right.cbegin(), right.cend(), left.cbegin(), left.cend(), std::back_inserter(output.only_right));
return output;
2016-11-07 12:12:54 +08:00
}
2017-03-30 08:33:08 +08:00
std::vector<T> only_left;
std::vector<T> both;
std::vector<T> only_right;
};
2017-03-30 09:14:48 +08:00
static std::vector<updated_port> find_updated_ports(const std::vector<std::string>& ports,
2017-04-04 07:13:46 +08:00
const std::map<std::string, VersionT>& previous_names_and_versions,
const std::map<std::string, VersionT>& current_names_and_versions)
2016-11-07 12:12:54 +08:00
{
2017-03-30 09:14:48 +08:00
std::vector<updated_port> output;
2017-03-30 08:33:08 +08:00
for (const std::string& name : ports)
2016-11-07 12:12:54 +08:00
{
2017-04-04 07:13:46 +08:00
const VersionT& previous_version = previous_names_and_versions.at(name);
const VersionT& current_version = current_names_and_versions.at(name);
2017-03-30 08:33:08 +08:00
if (previous_version == current_version)
{
continue;
}
output.push_back({ name, version_diff_t(previous_version, current_version) });
}
return output;
}
2017-04-04 07:13:46 +08:00
static void do_print_name_and_version(const std::vector<std::string>& ports_to_print, const std::map<std::string, VersionT>& names_and_versions)
2017-03-30 08:33:08 +08:00
{
for (const std::string& name : ports_to_print)
{
2017-04-04 07:13:46 +08:00
const VersionT& version = names_and_versions.at(name);
2017-03-30 08:33:08 +08:00
System::println("%-20s %-16s", name, version);
2016-11-07 12:12:54 +08:00
}
}
2017-04-04 07:13:46 +08:00
static std::map<std::string, VersionT> read_ports_from_commit(const vcpkg_paths& paths, const std::wstring& git_commit_id)
2016-11-07 12:12:54 +08:00
{
2017-03-11 08:34:24 +08:00
const fs::path& git_exe = paths.get_git_exe();
2016-11-07 12:12:54 +08:00
const fs::path dot_git_dir = paths.root / ".git";
const std::wstring ports_dir_name_as_string = paths.ports.filename().native();
const fs::path temp_checkout_path = paths.root / Strings::wformat(L"%s-%s", ports_dir_name_as_string, git_commit_id);
fs::create_directory(temp_checkout_path);
const std::wstring checkout_this_dir = Strings::wformat(LR"(.\%s)", ports_dir_name_as_string); // Must be relative to the root of the repository
2017-03-11 08:34:24 +08:00
const std::wstring cmd = Strings::wformat(LR"("%s" --git-dir="%s" --work-tree="%s" checkout %s -f -q -- %s %s & "%s" reset >NUL)",
git_exe.native(),
2016-11-07 12:12:54 +08:00
dot_git_dir.native(),
temp_checkout_path.native(),
git_commit_id,
checkout_this_dir,
2017-03-11 08:34:24 +08:00
L".vcpkg-root",
git_exe.native());
System::cmd_execute_clean(cmd);
const std::vector<SourceParagraph> source_paragraphs = Paragraphs::load_all_ports(temp_checkout_path / ports_dir_name_as_string);
2017-04-04 07:13:46 +08:00
const std::map<std::string, VersionT> names_and_versions = Paragraphs::extract_port_names_and_versions(source_paragraphs);
2016-11-07 12:12:54 +08:00
fs::remove_all(temp_checkout_path);
return names_and_versions;
}
2017-03-11 08:34:24 +08:00
static void check_commit_exists(const fs::path& git_exe, const std::wstring& git_commit_id)
{
static const std::string VALID_COMMIT_OUTPUT = "commit\n";
const std::wstring cmd = Strings::wformat(LR"("%s" cat-file -t %s)", git_exe.native(), git_commit_id);
const System::exit_code_and_output output = System::cmd_execute_and_capture_output(cmd);
Checks::check_exit(VCPKG_LINE_INFO, output.output == VALID_COMMIT_OUTPUT, "Invalid commit id %s", Strings::utf16_to_utf8(git_commit_id));
}
void perform_and_exit(const VcpkgCmdArguments& args, const vcpkg_paths& paths)
2016-11-07 12:12:54 +08:00
{
static const std::string example = Strings::format("The argument should be a branch/tag/hash to checkout.\n%s", Commands::Help::create_example_string("portsdiff mybranchname"));
args.check_min_arg_count(1, example);
args.check_max_arg_count(2, example);
2017-02-18 07:46:35 +08:00
args.check_and_get_optional_command_arguments({});
2017-03-11 08:34:24 +08:00
const fs::path& git_exe = paths.get_git_exe();
2016-11-07 12:12:54 +08:00
const std::wstring git_commit_id_for_previous_snapshot = Strings::utf8_to_utf16(args.command_arguments.at(0));
const std::wstring git_commit_id_for_current_snapshot = args.command_arguments.size() < 2 ? L"HEAD" : Strings::utf8_to_utf16(args.command_arguments.at(1));
2017-03-11 08:34:24 +08:00
check_commit_exists(git_exe, git_commit_id_for_current_snapshot);
check_commit_exists(git_exe, git_commit_id_for_previous_snapshot);
2017-04-04 07:13:46 +08:00
const std::map<std::string, VersionT> current_names_and_versions = read_ports_from_commit(paths, git_commit_id_for_current_snapshot);
const std::map<std::string, VersionT> previous_names_and_versions = read_ports_from_commit(paths, git_commit_id_for_previous_snapshot);
2016-11-07 12:12:54 +08:00
// Already sorted, so set_difference can work on std::vector too
std::vector<std::string> current_ports = Maps::extract_keys(current_names_and_versions);
std::vector<std::string> previous_ports = Maps::extract_keys(previous_names_and_versions);
2017-03-30 08:33:08 +08:00
const set_element_presence<std::string> setp = set_element_presence<std::string>::create(current_ports, previous_ports);
2016-11-07 12:12:54 +08:00
2017-03-30 08:33:08 +08:00
const std::vector<std::string>& added_ports = setp.only_left;
2016-11-07 12:12:54 +08:00
if (!added_ports.empty())
{
System::println("\nThe following %d ports were added:\n", added_ports.size());
do_print_name_and_version(added_ports, current_names_and_versions);
}
2017-03-30 08:33:08 +08:00
const std::vector<std::string>& removed_ports = setp.only_right;
2016-11-07 12:12:54 +08:00
if (!removed_ports.empty())
{
System::println("\nThe following %d ports were removed:\n", removed_ports.size());
do_print_name_and_version(removed_ports, previous_names_and_versions);
}
2017-03-30 08:33:08 +08:00
const std::vector<std::string>& common_ports = setp.both;
2017-03-30 09:14:48 +08:00
const std::vector<updated_port> updated_ports = find_updated_ports(common_ports, previous_names_and_versions, current_names_and_versions);
2016-11-07 12:12:54 +08:00
if (!updated_ports.empty())
{
System::println("\nThe following %d ports were updated:\n", updated_ports.size());
2017-03-30 09:14:48 +08:00
for (const updated_port& p : updated_ports)
2017-03-30 08:33:08 +08:00
{
2017-03-30 09:14:48 +08:00
System::println("%-20s %-16s", p.port, p.version_diff.toString());
2017-03-30 08:33:08 +08:00
}
2016-11-07 12:12:54 +08:00
}
if (added_ports.empty() && removed_ports.empty() && updated_ports.empty())
{
System::println("There were no changes in the ports between the two commits.");
}
Checks::exit_success(VCPKG_LINE_INFO);
2016-11-07 12:12:54 +08:00
}
}