CMake/Source/CTest/cmCTestResourceSpec.cxx
2019-11-25 11:08:27 -05:00

160 lines
3.9 KiB
C++

/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
file Copyright.txt or https://cmake.org/licensing for details. */
#include "cmCTestResourceSpec.h"
#include <map>
#include <string>
#include <utility>
#include <vector>
#include "cmsys/FStream.hxx"
#include "cmsys/RegularExpression.hxx"
#include "cm_jsoncpp_reader.h"
#include "cm_jsoncpp_value.h"
static const cmsys::RegularExpression IdentifierRegex{ "^[a-z_][a-z0-9_]*$" };
static const cmsys::RegularExpression IdRegex{ "^[a-z0-9_]+$" };
bool cmCTestResourceSpec::ReadFromJSONFile(const std::string& filename)
{
cmsys::ifstream fin(filename.c_str());
if (!fin) {
return false;
}
Json::Value root;
Json::CharReaderBuilder builder;
if (!Json::parseFromStream(builder, fin, &root, nullptr)) {
return false;
}
if (!root.isObject()) {
return false;
}
int majorVersion = 1;
int minorVersion = 0;
if (root.isMember("version")) {
auto const& version = root["version"];
if (version.isObject()) {
if (!version.isMember("major") || !version.isMember("minor")) {
return false;
}
auto const& major = version["major"];
auto const& minor = version["minor"];
if (!major.isInt() || !minor.isInt()) {
return false;
}
majorVersion = major.asInt();
minorVersion = minor.asInt();
} else {
return false;
}
} else {
return false;
}
if (majorVersion != 1 || minorVersion != 0) {
return false;
}
auto const& local = root["local"];
if (!local.isArray()) {
return false;
}
if (local.size() > 1) {
return false;
}
if (local.empty()) {
this->LocalSocket.Resources.clear();
return true;
}
auto const& localSocket = local[0];
if (!localSocket.isObject()) {
return false;
}
std::map<std::string, std::vector<cmCTestResourceSpec::Resource>> resources;
cmsys::RegularExpressionMatch match;
for (auto const& key : localSocket.getMemberNames()) {
if (IdentifierRegex.find(key.c_str(), match)) {
auto const& value = localSocket[key];
auto& r = resources[key];
if (value.isArray()) {
for (auto const& item : value) {
if (item.isObject()) {
cmCTestResourceSpec::Resource resource;
if (!item.isMember("id")) {
return false;
}
auto const& id = item["id"];
if (!id.isString()) {
return false;
}
resource.Id = id.asString();
if (!IdRegex.find(resource.Id.c_str(), match)) {
return false;
}
if (item.isMember("slots")) {
auto const& capacity = item["slots"];
if (!capacity.isConvertibleTo(Json::uintValue)) {
return false;
}
resource.Capacity = capacity.asUInt();
} else {
resource.Capacity = 1;
}
r.push_back(resource);
} else {
return false;
}
}
} else {
return false;
}
}
}
this->LocalSocket.Resources = std::move(resources);
return true;
}
bool cmCTestResourceSpec::operator==(const cmCTestResourceSpec& other) const
{
return this->LocalSocket == other.LocalSocket;
}
bool cmCTestResourceSpec::operator!=(const cmCTestResourceSpec& other) const
{
return !(*this == other);
}
bool cmCTestResourceSpec::Socket::operator==(
const cmCTestResourceSpec::Socket& other) const
{
return this->Resources == other.Resources;
}
bool cmCTestResourceSpec::Socket::operator!=(
const cmCTestResourceSpec::Socket& other) const
{
return !(*this == other);
}
bool cmCTestResourceSpec::Resource::operator==(
const cmCTestResourceSpec::Resource& other) const
{
return this->Id == other.Id && this->Capacity == other.Capacity;
}
bool cmCTestResourceSpec::Resource::operator!=(
const cmCTestResourceSpec::Resource& other) const
{
return !(*this == other);
}