/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying file Copyright.txt or https://cmake.org/licensing for details. */ #include "cmFileAPI.h" #include "cmAlgorithms.h" #include "cmCryptoHash.h" #include "cmFileAPICache.h" #include "cmFileAPICodemodel.h" #include "cmGlobalGenerator.h" #include "cmSystemTools.h" #include "cmTimestamp.h" #include "cmake.h" #include "cmsys/Directory.hxx" #include "cmsys/FStream.hxx" #include #include #include #include #include #include #include cmFileAPI::cmFileAPI(cmake* cm) : CMakeInstance(cm) { this->APIv1 = this->CMakeInstance->GetHomeOutputDirectory() + "/.cmake/api/v1"; Json::CharReaderBuilder rbuilder; rbuilder["collectComments"] = false; rbuilder["failIfExtra"] = true; rbuilder["rejectDupKeys"] = false; rbuilder["strictRoot"] = true; this->JsonReader = std::unique_ptr(rbuilder.newCharReader()); Json::StreamWriterBuilder wbuilder; wbuilder["indentation"] = "\t"; this->JsonWriter = std::unique_ptr(wbuilder.newStreamWriter()); } void cmFileAPI::ReadQueries() { std::string const query_dir = this->APIv1 + "/query"; this->QueryExists = cmSystemTools::FileIsDirectory(query_dir); if (!this->QueryExists) { return; } // Load queries at the top level. std::vector queries = cmFileAPI::LoadDir(query_dir); // Read the queries and save for later. for (std::string& query : queries) { if (cmHasLiteralPrefix(query, "client-")) { this->ReadClient(query); } else if (!cmFileAPI::ReadQuery(query, this->TopQuery.Known)) { this->TopQuery.Unknown.push_back(std::move(query)); } } } void cmFileAPI::WriteReplies() { if (this->QueryExists) { cmSystemTools::MakeDirectory(this->APIv1 + "/reply"); this->WriteJsonFile(this->BuildReplyIndex(), "index", ComputeSuffixTime); } this->RemoveOldReplyFiles(); } std::vector cmFileAPI::LoadDir(std::string const& dir) { std::vector files; cmsys::Directory d; d.Load(dir); for (unsigned long i = 0; i < d.GetNumberOfFiles(); ++i) { std::string f = d.GetFile(i); if (f != "." && f != "..") { files.push_back(std::move(f)); } } std::sort(files.begin(), files.end()); return files; } void cmFileAPI::RemoveOldReplyFiles() { std::string const reply_dir = this->APIv1 + "/reply"; std::vector files = this->LoadDir(reply_dir); for (std::string const& f : files) { if (this->ReplyFiles.find(f) == this->ReplyFiles.end()) { std::string file = reply_dir + "/" + f; cmSystemTools::RemoveFile(file); } } } bool cmFileAPI::ReadJsonFile(std::string const& file, Json::Value& value, std::string& error) { std::vector content; cmsys::ifstream fin; if (!cmSystemTools::FileIsDirectory(file)) { fin.open(file.c_str(), std::ios::binary); } auto finEnd = fin.rdbuf()->pubseekoff(0, std::ios::end); if (finEnd > 0) { size_t finSize = finEnd; try { // Allocate a buffer to read the whole file. content.resize(finSize); // Now read the file from the beginning. fin.seekg(0, std::ios::beg); fin.read(content.data(), finSize); } catch (...) { fin.setstate(std::ios::failbit); } } fin.close(); if (!fin) { value = Json::Value(); error = "failed to read from file"; return false; } // Parse our buffer as json. if (!this->JsonReader->parse(content.data(), content.data() + content.size(), &value, &error)) { value = Json::Value(); return false; } return true; } std::string cmFileAPI::WriteJsonFile( Json::Value const& value, std::string const& prefix, std::string (*computeSuffix)(std::string const&)) { std::string fileName; // Write the json file with a temporary name. std::string const& tmpFile = this->APIv1 + "/tmp.json"; cmsys::ofstream ftmp(tmpFile.c_str()); this->JsonWriter->write(value, &ftmp); ftmp << "\n"; ftmp.close(); if (!ftmp) { cmSystemTools::RemoveFile(tmpFile); return fileName; } // Compute the final name for the file. fileName = prefix + "-" + computeSuffix(tmpFile) + ".json"; // Create the destination. std::string file = this->APIv1 + "/reply"; cmSystemTools::MakeDirectory(file); file += "/"; file += fileName; // If the final name already exists then assume it has proper content. // Otherwise, atomically place the reply file at its final name if (cmSystemTools::FileExists(file, true) || !cmSystemTools::RenameFile(tmpFile.c_str(), file.c_str())) { cmSystemTools::RemoveFile(tmpFile); } // Record this among files we have just written. this->ReplyFiles.insert(fileName); return fileName; } Json::Value cmFileAPI::MaybeJsonFile(Json::Value in, std::string const& prefix) { Json::Value out; if (in.isObject() || in.isArray()) { out = Json::objectValue; out["jsonFile"] = this->WriteJsonFile(in, prefix); } else { out = std::move(in); } return out; } std::string cmFileAPI::ComputeSuffixHash(std::string const& file) { cmCryptoHash hasher(cmCryptoHash::AlgoSHA3_256); std::string hash = hasher.HashFile(file); hash.resize(20, '0'); return hash; } std::string cmFileAPI::ComputeSuffixTime(std::string const&) { std::chrono::milliseconds ms = std::chrono::duration_cast( std::chrono::system_clock::now().time_since_epoch()); std::chrono::seconds s = std::chrono::duration_cast(ms); std::time_t ts = s.count(); std::size_t tms = ms.count() % 1000; cmTimestamp cmts; std::ostringstream ss; ss << cmts.CreateTimestampFromTimeT(ts, "%Y-%m-%dT%H-%M-%S", true) << '-' << std::setfill('0') << std::setw(4) << tms; return ss.str(); } bool cmFileAPI::ReadQuery(std::string const& query, std::vector& objects) { // Parse the "-" syntax. std::string::size_type sep_pos = query.find('-'); if (sep_pos == std::string::npos) { return false; } std::string kindName = query.substr(0, sep_pos); std::string verStr = query.substr(sep_pos + 1); if (kindName == ObjectKindName(ObjectKind::CodeModel)) { Object o; o.Kind = ObjectKind::CodeModel; if (verStr == "v2") { o.Version = 2; } else { return false; } objects.push_back(o); return true; } if (kindName == ObjectKindName(ObjectKind::Cache)) { Object o; o.Kind = ObjectKind::Cache; if (verStr == "v2") { o.Version = 2; } else { return false; } objects.push_back(o); return true; } if (kindName == ObjectKindName(ObjectKind::InternalTest)) { Object o; o.Kind = ObjectKind::InternalTest; if (verStr == "v1") { o.Version = 1; } else if (verStr == "v2") { o.Version = 2; } else { return false; } objects.push_back(o); return true; } return false; } void cmFileAPI::ReadClient(std::string const& client) { // Load queries for the client. std::string clientDir = this->APIv1 + "/query/" + client; std::vector queries = this->LoadDir(clientDir); // Read the queries and save for later. ClientQuery& clientQuery = this->ClientQueries[client]; for (std::string& query : queries) { if (query == "query.json") { clientQuery.HaveQueryJson = true; this->ReadClientQuery(client, clientQuery.QueryJson); } else if (!this->ReadQuery(query, clientQuery.DirQuery.Known)) { clientQuery.DirQuery.Unknown.push_back(std::move(query)); } } } void cmFileAPI::ReadClientQuery(std::string const& client, ClientQueryJson& q) { // Read the query.json file. std::string queryFile = this->APIv1 + "/query/" + client + "/query.json"; Json::Value query; if (!this->ReadJsonFile(queryFile, query, q.Error)) { return; } if (!query.isObject()) { q.Error = "query root is not an object"; return; } Json::Value const& clientValue = query["client"]; if (!clientValue.isNull()) { q.ClientValue = clientValue; } q.RequestsValue = std::move(query["requests"]); q.Requests = this->BuildClientRequests(q.RequestsValue); } Json::Value cmFileAPI::BuildReplyIndex() { Json::Value index(Json::objectValue); // Report information about this version of CMake. index["cmake"] = this->BuildCMake(); // Reply to all queries that we loaded. Json::Value& reply = index["reply"] = this->BuildReply(this->TopQuery); for (auto const& client : this->ClientQueries) { std::string const& clientName = client.first; ClientQuery const& clientQuery = client.second; reply[clientName] = this->BuildClientReply(clientQuery); } // Move our index of generated objects into its field. Json::Value& objects = index["objects"] = Json::arrayValue; for (auto& entry : this->ReplyIndexObjects) { objects.append(std::move(entry.second)); // NOLINT(*) } return index; } Json::Value cmFileAPI::BuildCMake() { Json::Value cmake = Json::objectValue; cmake["version"] = this->CMakeInstance->ReportVersionJson(); Json::Value& cmake_paths = cmake["paths"] = Json::objectValue; cmake_paths["cmake"] = cmSystemTools::GetCMakeCommand(); cmake_paths["ctest"] = cmSystemTools::GetCTestCommand(); cmake_paths["cpack"] = cmSystemTools::GetCPackCommand(); cmake_paths["root"] = cmSystemTools::GetCMakeRoot(); cmake["generator"] = this->CMakeInstance->GetGlobalGenerator()->GetJson(); return cmake; } Json::Value cmFileAPI::BuildReply(Query const& q) { Json::Value reply = Json::objectValue; for (Object const& o : q.Known) { std::string const& name = ObjectName(o); reply[name] = this->AddReplyIndexObject(o); } for (std::string const& name : q.Unknown) { reply[name] = cmFileAPI::BuildReplyError("unknown query file"); } return reply; } Json::Value cmFileAPI::BuildReplyError(std::string const& error) { Json::Value e = Json::objectValue; e["error"] = error; return e; } Json::Value const& cmFileAPI::AddReplyIndexObject(Object const& o) { Json::Value& indexEntry = this->ReplyIndexObjects[o]; if (!indexEntry.isNull()) { // The reply object has already been generated. return indexEntry; } // Generate this reply object. Json::Value const& object = this->BuildObject(o); assert(object.isObject()); // Populate this index entry. indexEntry = Json::objectValue; indexEntry["kind"] = object["kind"]; indexEntry["version"] = object["version"]; indexEntry["jsonFile"] = this->WriteJsonFile(object, ObjectName(o)); return indexEntry; } const char* cmFileAPI::ObjectKindName(ObjectKind kind) { // Keep in sync with ObjectKind enum. static const char* objectKindNames[] = { "codemodel", // "cache", // "__test" // }; return objectKindNames[size_t(kind)]; } std::string cmFileAPI::ObjectName(Object const& o) { std::string name = ObjectKindName(o.Kind); name += "-v"; name += std::to_string(o.Version); return name; } Json::Value cmFileAPI::BuildObject(Object const& object) { Json::Value value; switch (object.Kind) { case ObjectKind::CodeModel: value = this->BuildCodeModel(object); break; case ObjectKind::Cache: value = this->BuildCache(object); break; case ObjectKind::InternalTest: value = this->BuildInternalTest(object); break; } return value; } cmFileAPI::ClientRequests cmFileAPI::BuildClientRequests( Json::Value const& requests) { ClientRequests result; if (requests.isNull()) { result.Error = "'requests' member missing"; return result; } if (!requests.isArray()) { result.Error = "'requests' member is not an array"; return result; } result.reserve(requests.size()); for (Json::Value const& request : requests) { result.emplace_back(this->BuildClientRequest(request)); } return result; } cmFileAPI::ClientRequest cmFileAPI::BuildClientRequest( Json::Value const& request) { ClientRequest r; if (!request.isObject()) { r.Error = "request is not an object"; return r; } Json::Value const& kind = request["kind"]; if (kind.isNull()) { r.Error = "'kind' member missing"; return r; } if (!kind.isString()) { r.Error = "'kind' member is not a string"; return r; } std::string const& kindName = kind.asString(); if (kindName == this->ObjectKindName(ObjectKind::CodeModel)) { r.Kind = ObjectKind::CodeModel; } else if (kindName == this->ObjectKindName(ObjectKind::Cache)) { r.Kind = ObjectKind::Cache; } else if (kindName == this->ObjectKindName(ObjectKind::InternalTest)) { r.Kind = ObjectKind::InternalTest; } else { r.Error = "unknown request kind '" + kindName + "'"; return r; } Json::Value const& version = request["version"]; if (version.isNull()) { r.Error = "'version' member missing"; return r; } std::vector versions; if (!cmFileAPI::ReadRequestVersions(version, versions, r.Error)) { return r; } switch (r.Kind) { case ObjectKind::CodeModel: this->BuildClientRequestCodeModel(r, versions); break; case ObjectKind::Cache: this->BuildClientRequestCache(r, versions); break; case ObjectKind::InternalTest: this->BuildClientRequestInternalTest(r, versions); break; } return r; } Json::Value cmFileAPI::BuildClientReply(ClientQuery const& q) { Json::Value reply = this->BuildReply(q.DirQuery); if (!q.HaveQueryJson) { return reply; } Json::Value& reply_query_json = reply["query.json"]; ClientQueryJson const& qj = q.QueryJson; if (!qj.Error.empty()) { reply_query_json = this->BuildReplyError(qj.Error); return reply; } if (!qj.ClientValue.isNull()) { reply_query_json["client"] = qj.ClientValue; } if (!qj.RequestsValue.isNull()) { reply_query_json["requests"] = qj.RequestsValue; } reply_query_json["responses"] = this->BuildClientReplyResponses(qj.Requests); return reply; } Json::Value cmFileAPI::BuildClientReplyResponses( ClientRequests const& requests) { Json::Value responses; if (!requests.Error.empty()) { responses = this->BuildReplyError(requests.Error); return responses; } responses = Json::arrayValue; for (ClientRequest const& request : requests) { responses.append(this->BuildClientReplyResponse(request)); } return responses; } Json::Value cmFileAPI::BuildClientReplyResponse(ClientRequest const& request) { Json::Value response; if (!request.Error.empty()) { response = this->BuildReplyError(request.Error); return response; } response = this->AddReplyIndexObject(request); return response; } bool cmFileAPI::ReadRequestVersions(Json::Value const& version, std::vector& versions, std::string& error) { if (version.isArray()) { for (Json::Value const& v : version) { if (!ReadRequestVersion(v, /*inArray=*/true, versions, error)) { return false; } } } else { if (!ReadRequestVersion(version, /*inArray=*/false, versions, error)) { return false; } } return true; } bool cmFileAPI::ReadRequestVersion(Json::Value const& version, bool inArray, std::vector& result, std::string& error) { if (version.isUInt()) { RequestVersion v; v.Major = version.asUInt(); result.push_back(v); return true; } if (!version.isObject()) { if (inArray) { error = "'version' array entry is not a non-negative integer or object"; } else { error = "'version' member is not a non-negative integer, object, or array"; } return false; } Json::Value const& major = version["major"]; if (major.isNull()) { error = "'version' object 'major' member missing"; return false; } if (!major.isUInt()) { error = "'version' object 'major' member is not a non-negative integer"; return false; } RequestVersion v; v.Major = major.asUInt(); Json::Value const& minor = version["minor"]; if (minor.isUInt()) { v.Minor = minor.asUInt(); } else if (!minor.isNull()) { error = "'version' object 'minor' member is not a non-negative integer"; return false; } result.push_back(v); return true; } std::string cmFileAPI::NoSupportedVersion( std::vector const& versions) { std::ostringstream msg; msg << "no supported version specified"; if (!versions.empty()) { msg << " among:"; for (RequestVersion const& v : versions) { msg << " " << v.Major << "." << v.Minor; } } return msg.str(); } // The "codemodel" object kind. static unsigned int const CodeModelV2Minor = 0; void cmFileAPI::BuildClientRequestCodeModel( ClientRequest& r, std::vector const& versions) { // Select a known version from those requested. for (RequestVersion const& v : versions) { if ((v.Major == 2 && v.Minor <= CodeModelV2Minor)) { r.Version = v.Major; break; } } if (!r.Version) { r.Error = NoSupportedVersion(versions); } } Json::Value cmFileAPI::BuildCodeModel(Object const& object) { using namespace std::placeholders; Json::Value codemodel = cmFileAPICodemodelDump(*this, object.Version); codemodel["kind"] = this->ObjectKindName(object.Kind); Json::Value& version = codemodel["version"] = Json::objectValue; if (object.Version == 2) { version["major"] = 2; version["minor"] = CodeModelV2Minor; } else { return codemodel; // should be unreachable } return codemodel; } // The "cache" object kind. static unsigned int const CacheV2Minor = 0; void cmFileAPI::BuildClientRequestCache( ClientRequest& r, std::vector const& versions) { // Select a known version from those requested. for (RequestVersion const& v : versions) { if ((v.Major == 2 && v.Minor <= CacheV2Minor)) { r.Version = v.Major; break; } } if (!r.Version) { r.Error = NoSupportedVersion(versions); } } Json::Value cmFileAPI::BuildCache(Object const& object) { using namespace std::placeholders; Json::Value cache = cmFileAPICacheDump(*this, object.Version); cache["kind"] = this->ObjectKindName(object.Kind); Json::Value& version = cache["version"] = Json::objectValue; if (object.Version == 2) { version["major"] = 2; version["minor"] = CacheV2Minor; } else { return cache; // should be unreachable } return cache; } // The "__test" object kind is for internal testing of CMake. static unsigned int const InternalTestV1Minor = 3; static unsigned int const InternalTestV2Minor = 0; void cmFileAPI::BuildClientRequestInternalTest( ClientRequest& r, std::vector const& versions) { // Select a known version from those requested. for (RequestVersion const& v : versions) { if ((v.Major == 1 && v.Minor <= InternalTestV1Minor) || // (v.Major == 2 && v.Minor <= InternalTestV2Minor)) { r.Version = v.Major; break; } } if (!r.Version) { r.Error = NoSupportedVersion(versions); } } Json::Value cmFileAPI::BuildInternalTest(Object const& object) { Json::Value test = Json::objectValue; test["kind"] = this->ObjectKindName(object.Kind); Json::Value& version = test["version"] = Json::objectValue; if (object.Version == 2) { version["major"] = 2; version["minor"] = InternalTestV2Minor; } else { version["major"] = 1; version["minor"] = InternalTestV1Minor; } return test; }