#include "vcpkg.h" #include #include #include #include #include #include #include #include #include "vcpkg_Files.h" #include "Paragraphs.h" #include using namespace vcpkg; static StatusParagraphs load_current_database(const fs::path& vcpkg_dir_status_file, const fs::path& vcpkg_dir_status_file_old) { if (!fs::exists(vcpkg_dir_status_file)) { if (!fs::exists(vcpkg_dir_status_file_old)) { // no status file, use empty db return StatusParagraphs(); } fs::rename(vcpkg_dir_status_file_old, vcpkg_dir_status_file); } auto text = Files::get_contents(vcpkg_dir_status_file).get_or_throw(); auto pghs = Paragraphs::parse_paragraphs(text); std::vector> status_pghs; for (auto&& p : pghs) { status_pghs.push_back(std::make_unique(p)); } return StatusParagraphs(std::move(status_pghs)); } StatusParagraphs vcpkg::database_load_check(const vcpkg_paths& paths) { auto updates_dir = paths.vcpkg_dir_updates; std::error_code ec; fs::create_directory(paths.installed, ec); fs::create_directory(paths.vcpkg_dir, ec); fs::create_directory(paths.vcpkg_dir_info, ec); fs::create_directory(updates_dir, ec); const fs::path& status_file = paths.vcpkg_dir_status_file; const fs::path status_file_old = status_file.parent_path() / "status-old"; const fs::path status_file_new = status_file.parent_path() / "status-new"; StatusParagraphs current_status_db = load_current_database(status_file, status_file_old); auto b = fs::directory_iterator(updates_dir); auto e = fs::directory_iterator(); if (b == e) { // updates directory is empty, control file is up-to-date. return current_status_db; } for (; b != e; ++b) { if (!fs::is_regular_file(b->status())) continue; if (b->path().filename() == "incomplete") continue; auto text = Files::get_contents(b->path()).get_or_throw(); auto pghs = Paragraphs::parse_paragraphs(text); for (auto&& p : pghs) { current_status_db.insert(std::make_unique(p)); } } std::fstream(status_file_new, std::ios_base::out | std::ios_base::binary | std::ios_base::trunc) << current_status_db; if (fs::exists(status_file_old)) fs::remove(status_file_old); if (fs::exists(status_file)) fs::rename(status_file, status_file_old); fs::rename(status_file_new, status_file); fs::remove(status_file_old); b = fs::directory_iterator(updates_dir); for (; b != e; ++b) { if (!fs::is_regular_file(b->status())) continue; fs::remove(b->path()); } return current_status_db; } void vcpkg::write_update(const vcpkg_paths& paths, const StatusParagraph& p) { static int update_id = 0; auto my_update_id = update_id++; auto tmp_update_filename = paths.vcpkg_dir_updates / "incomplete"; auto update_filename = paths.vcpkg_dir_updates / std::to_string(my_update_id); std::fstream fs(tmp_update_filename, std::ios_base::out | std::ios_base::binary | std::ios_base::trunc); fs << p; fs.close(); fs::rename(tmp_update_filename, update_filename); } std::vector vcpkg::get_installed_files(const vcpkg_paths& paths, const StatusParagraphs& status_db) { static const std::string MARK_FOR_REMOVAL = ""; std::vector installed_files; std::string line; for (const std::unique_ptr& pgh : status_db) { if (pgh->state != install_state_t::installed) { continue; } std::fstream listfile(paths.listfile_path(pgh->package)); std::vector installed_files_of_current_pgh; while (std::getline(listfile, line)) { if (line.empty()) { continue; } installed_files_of_current_pgh.push_back(line); } // Should already be sorted std::sort(installed_files_of_current_pgh.begin(), installed_files_of_current_pgh.end()); // Since the files are sorted, we can detect the entries that represent directories // by comparing every element with the next one and checking if the next has a slash immediately after the current one's length for (int i = 1; i < installed_files_of_current_pgh.size(); i++) { std::string& current_string = installed_files_of_current_pgh.at(i - 1); const std::string& next_string = installed_files_of_current_pgh.at(i); const size_t potential_slash_char_index = current_string.length(); // Make sure the index exists first if (next_string.size() > potential_slash_char_index && next_string.at(potential_slash_char_index) == '/') { current_string = MARK_FOR_REMOVAL; } } installed_files_of_current_pgh.erase(std::remove_if(installed_files_of_current_pgh.begin(), installed_files_of_current_pgh.end(), [](const std::string& file) { return file == MARK_FOR_REMOVAL; }), installed_files_of_current_pgh.end()); const StatusParagraph_and_associated_files pgh_and_files = {*pgh, std::move(installed_files_of_current_pgh)}; installed_files.push_back(pgh_and_files); } return installed_files; } expected vcpkg::try_load_port(const fs::path& path) { try { auto pghs = Paragraphs::get_paragraphs(path / "CONTROL"); Checks::check_exit(pghs.size() == 1, "Invalid control file at %s\\CONTROL", path.string()); return SourceParagraph(pghs[0]); } catch (std::runtime_error const&) { } return std::errc::no_such_file_or_directory; } expected vcpkg::try_load_cached_package(const vcpkg_paths& paths, const package_spec& spec) { const fs::path path = paths.package_dir(spec) / "CONTROL"; auto control_contents_maybe = Files::get_contents(path); if (auto control_contents = control_contents_maybe.get()) { std::vector> pghs; try { pghs = Paragraphs::parse_paragraphs(*control_contents); } catch (std::runtime_error) { } Checks::check_exit(pghs.size() == 1, "Invalid control file at %s", path.string()); return BinaryParagraph(pghs[0]); } return control_contents_maybe.error_code(); }