cmake/Source/cmCMakePresetsGraphReadJSON.cxx

787 lines
26 KiB
C++
Raw Normal View History

2022-03-29 21:10:50 +02:00
/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
file Copyright.txt or https://cmake.org/licensing for details. */
#include <algorithm>
2023-07-02 19:51:09 +02:00
#include <fstream>
2022-03-29 21:10:50 +02:00
#include <functional>
#include <map>
#include <string>
#include <unordered_set>
#include <utility>
#include <vector>
#include <cm/memory>
#include <cm/optional>
#include <cmext/string_view>
#include <cm3p/json/value.h>
2023-12-07 09:12:54 +01:00
#include "cmCMakePresetsErrors.h"
2022-03-29 21:10:50 +02:00
#include "cmCMakePresetsGraph.h"
#include "cmCMakePresetsGraphInternal.h"
#include "cmJSONHelpers.h"
2023-07-02 19:51:09 +02:00
#include "cmJSONState.h"
2022-03-29 21:10:50 +02:00
#include "cmStringAlgorithms.h"
#include "cmSystemTools.h"
#include "cmVersion.h"
namespace {
using CacheVariable = cmCMakePresetsGraph::CacheVariable;
using ConfigurePreset = cmCMakePresetsGraph::ConfigurePreset;
using BuildPreset = cmCMakePresetsGraph::BuildPreset;
using TestPreset = cmCMakePresetsGraph::TestPreset;
2022-11-16 20:14:03 +01:00
using PackagePreset = cmCMakePresetsGraph::PackagePreset;
using WorkflowPreset = cmCMakePresetsGraph::WorkflowPreset;
2022-03-29 21:10:50 +02:00
using ArchToolsetStrategy = cmCMakePresetsGraph::ArchToolsetStrategy;
2023-07-02 19:51:09 +02:00
using JSONHelperBuilder = cmJSONHelperBuilder;
using ExpandMacroResult = cmCMakePresetsGraphInternal::ExpandMacroResult;
using MacroExpander = cmCMakePresetsGraphInternal::MacroExpander;
2024-07-09 14:46:46 +02:00
using MacroExpanderVector = cmCMakePresetsGraphInternal::MacroExpanderVector;
using cmCMakePresetsGraphInternal::BaseMacroExpander;
2023-07-02 19:51:09 +02:00
using cmCMakePresetsGraphInternal::ExpandMacros;
2022-03-29 21:10:50 +02:00
constexpr int MIN_VERSION = 1;
2024-11-11 15:18:55 +01:00
constexpr int MAX_VERSION = 10;
2022-03-29 21:10:50 +02:00
struct CMakeVersion
{
unsigned int Major = 0;
unsigned int Minor = 0;
unsigned int Patch = 0;
};
struct RootPresets
{
CMakeVersion CMakeMinimumRequired;
2022-11-16 20:14:03 +01:00
std::vector<ConfigurePreset> ConfigurePresets;
std::vector<BuildPreset> BuildPresets;
std::vector<TestPreset> TestPresets;
std::vector<PackagePreset> PackagePresets;
std::vector<WorkflowPreset> WorkflowPresets;
2022-03-29 21:10:50 +02:00
std::vector<std::string> Include;
};
std::unique_ptr<cmCMakePresetsGraphInternal::NotCondition> InvertCondition(
std::unique_ptr<cmCMakePresetsGraph::Condition> condition)
{
auto retval = cm::make_unique<cmCMakePresetsGraphInternal::NotCondition>();
retval->SubCondition = std::move(condition);
return retval;
}
2023-07-02 19:51:09 +02:00
auto const ConditionStringHelper = JSONHelperBuilder::String();
2022-03-29 21:10:50 +02:00
2023-07-02 19:51:09 +02:00
auto const ConditionBoolHelper = JSONHelperBuilder::Bool();
2022-03-29 21:10:50 +02:00
2022-08-04 22:12:04 +02:00
auto const ConditionStringListHelper = JSONHelperBuilder::Vector<std::string>(
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_CONDITION, ConditionStringHelper);
2022-03-29 21:10:50 +02:00
auto const ConstConditionHelper =
2022-08-04 22:12:04 +02:00
JSONHelperBuilder::Object<cmCMakePresetsGraphInternal::ConstCondition>(
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_CONDITION_OBJECT, false)
2022-03-29 21:10:50 +02:00
.Bind<std::string>("type"_s, nullptr, ConditionStringHelper, true)
.Bind("value"_s, &cmCMakePresetsGraphInternal::ConstCondition::Value,
ConditionBoolHelper, true);
auto const EqualsConditionHelper =
2022-08-04 22:12:04 +02:00
JSONHelperBuilder::Object<cmCMakePresetsGraphInternal::EqualsCondition>(
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_CONDITION_OBJECT, false)
2022-03-29 21:10:50 +02:00
.Bind<std::string>("type"_s, nullptr, ConditionStringHelper, true)
.Bind("lhs"_s, &cmCMakePresetsGraphInternal::EqualsCondition::Lhs,
ConditionStringHelper, true)
.Bind("rhs"_s, &cmCMakePresetsGraphInternal::EqualsCondition::Rhs,
ConditionStringHelper, true);
auto const InListConditionHelper =
2022-08-04 22:12:04 +02:00
JSONHelperBuilder::Object<cmCMakePresetsGraphInternal::InListCondition>(
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_CONDITION_OBJECT, false)
2022-03-29 21:10:50 +02:00
.Bind<std::string>("type"_s, nullptr, ConditionStringHelper, true)
.Bind("string"_s, &cmCMakePresetsGraphInternal::InListCondition::String,
ConditionStringHelper, true)
.Bind("list"_s, &cmCMakePresetsGraphInternal::InListCondition::List,
ConditionStringListHelper, true);
auto const MatchesConditionHelper =
2022-08-04 22:12:04 +02:00
JSONHelperBuilder::Object<cmCMakePresetsGraphInternal::MatchesCondition>(
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_CONDITION_OBJECT, false)
2022-03-29 21:10:50 +02:00
.Bind<std::string>("type"_s, nullptr, ConditionStringHelper, true)
.Bind("string"_s, &cmCMakePresetsGraphInternal::MatchesCondition::String,
ConditionStringHelper, true)
.Bind("regex"_s, &cmCMakePresetsGraphInternal::MatchesCondition::Regex,
ConditionStringHelper, true);
2023-07-02 19:51:09 +02:00
bool SubConditionHelper(std::unique_ptr<cmCMakePresetsGraph::Condition>& out,
const Json::Value* value, cmJSONState* state);
2022-03-29 21:10:50 +02:00
auto const ListConditionVectorHelper =
2022-08-04 22:12:04 +02:00
JSONHelperBuilder::Vector<std::unique_ptr<cmCMakePresetsGraph::Condition>>(
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_CONDITION, SubConditionHelper);
2022-03-29 21:10:50 +02:00
auto const AnyAllOfConditionHelper =
2022-08-04 22:12:04 +02:00
JSONHelperBuilder::Object<cmCMakePresetsGraphInternal::AnyAllOfCondition>(
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_CONDITION_OBJECT, false)
2022-03-29 21:10:50 +02:00
.Bind<std::string>("type"_s, nullptr, ConditionStringHelper, true)
.Bind("conditions"_s,
&cmCMakePresetsGraphInternal::AnyAllOfCondition::Conditions,
ListConditionVectorHelper);
auto const NotConditionHelper =
2022-08-04 22:12:04 +02:00
JSONHelperBuilder::Object<cmCMakePresetsGraphInternal::NotCondition>(
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_CONDITION_OBJECT, false)
2022-03-29 21:10:50 +02:00
.Bind<std::string>("type"_s, nullptr, ConditionStringHelper, true)
.Bind("condition"_s,
&cmCMakePresetsGraphInternal::NotCondition::SubCondition,
SubConditionHelper);
2023-07-02 19:51:09 +02:00
bool ConditionHelper(std::unique_ptr<cmCMakePresetsGraph::Condition>& out,
const Json::Value* value, cmJSONState* state)
2022-03-29 21:10:50 +02:00
{
if (!value) {
out.reset();
2023-07-02 19:51:09 +02:00
return true;
2022-03-29 21:10:50 +02:00
}
if (value->isBool()) {
auto c = cm::make_unique<cmCMakePresetsGraphInternal::ConstCondition>();
c->Value = value->asBool();
out = std::move(c);
2023-07-02 19:51:09 +02:00
return true;
2022-03-29 21:10:50 +02:00
}
if (value->isNull()) {
out = cm::make_unique<cmCMakePresetsGraphInternal::NullCondition>();
2023-07-02 19:51:09 +02:00
return true;
2022-03-29 21:10:50 +02:00
}
if (value->isObject()) {
if (!value->isMember("type")) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_CONDITION(value, state);
2023-07-02 19:51:09 +02:00
return false;
2022-03-29 21:10:50 +02:00
}
if (!(*value)["type"].isString()) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_CONDITION(value, state);
2023-07-02 19:51:09 +02:00
return false;
2022-03-29 21:10:50 +02:00
}
auto type = (*value)["type"].asString();
if (type == "const") {
auto c = cm::make_unique<cmCMakePresetsGraphInternal::ConstCondition>();
2023-07-02 19:51:09 +02:00
CHECK_OK(ConstConditionHelper(*c, value, state));
2022-03-29 21:10:50 +02:00
out = std::move(c);
2023-07-02 19:51:09 +02:00
return true;
2022-03-29 21:10:50 +02:00
}
if (type == "equals" || type == "notEquals") {
auto c = cm::make_unique<cmCMakePresetsGraphInternal::EqualsCondition>();
2023-07-02 19:51:09 +02:00
CHECK_OK(EqualsConditionHelper(*c, value, state));
2022-03-29 21:10:50 +02:00
out = std::move(c);
if (type == "notEquals") {
out = InvertCondition(std::move(out));
}
2023-07-02 19:51:09 +02:00
return true;
2022-03-29 21:10:50 +02:00
}
if (type == "inList" || type == "notInList") {
auto c = cm::make_unique<cmCMakePresetsGraphInternal::InListCondition>();
2023-07-02 19:51:09 +02:00
CHECK_OK(InListConditionHelper(*c, value, state));
2022-03-29 21:10:50 +02:00
out = std::move(c);
if (type == "notInList") {
out = InvertCondition(std::move(out));
}
2023-07-02 19:51:09 +02:00
return true;
2022-03-29 21:10:50 +02:00
}
if (type == "matches" || type == "notMatches") {
auto c =
cm::make_unique<cmCMakePresetsGraphInternal::MatchesCondition>();
2023-07-02 19:51:09 +02:00
CHECK_OK(MatchesConditionHelper(*c, value, state));
2022-03-29 21:10:50 +02:00
out = std::move(c);
if (type == "notMatches") {
out = InvertCondition(std::move(out));
}
2023-07-02 19:51:09 +02:00
return true;
2022-03-29 21:10:50 +02:00
}
if (type == "anyOf" || type == "allOf") {
auto c =
cm::make_unique<cmCMakePresetsGraphInternal::AnyAllOfCondition>();
c->StopValue = (type == "anyOf");
2023-07-02 19:51:09 +02:00
CHECK_OK(AnyAllOfConditionHelper(*c, value, state));
2022-03-29 21:10:50 +02:00
out = std::move(c);
2023-07-02 19:51:09 +02:00
return true;
2022-03-29 21:10:50 +02:00
}
if (type == "not") {
auto c = cm::make_unique<cmCMakePresetsGraphInternal::NotCondition>();
2023-07-02 19:51:09 +02:00
CHECK_OK(NotConditionHelper(*c, value, state));
2022-03-29 21:10:50 +02:00
out = std::move(c);
2023-07-02 19:51:09 +02:00
return true;
2022-03-29 21:10:50 +02:00
}
}
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_CONDITION(value, state);
2023-07-02 19:51:09 +02:00
return false;
2022-03-29 21:10:50 +02:00
}
2023-07-02 19:51:09 +02:00
bool SubConditionHelper(std::unique_ptr<cmCMakePresetsGraph::Condition>& out,
const Json::Value* value, cmJSONState* state)
2022-03-29 21:10:50 +02:00
{
std::unique_ptr<cmCMakePresetsGraph::Condition> ptr;
2023-07-02 19:51:09 +02:00
auto result = ConditionHelper(ptr, value, state);
2022-03-29 21:10:50 +02:00
if (ptr && ptr->IsNull()) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_CONDITION(value, state);
2023-07-02 19:51:09 +02:00
return false;
2022-03-29 21:10:50 +02:00
}
out = std::move(ptr);
return result;
}
2023-07-02 19:51:09 +02:00
bool EnvironmentHelper(cm::optional<std::string>& out,
const Json::Value* value, cmJSONState* state)
2022-03-29 21:10:50 +02:00
{
if (!value || value->isNull()) {
out = cm::nullopt;
2023-07-02 19:51:09 +02:00
return true;
2022-03-29 21:10:50 +02:00
}
if (value->isString()) {
out = value->asString();
2023-07-02 19:51:09 +02:00
return true;
2022-03-29 21:10:50 +02:00
}
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_PRESET(value, state);
2023-07-02 19:51:09 +02:00
return false;
2022-03-29 21:10:50 +02:00
}
2023-07-02 19:51:09 +02:00
auto const VersionIntHelper =
2023-12-07 09:12:54 +01:00
JSONHelperBuilder::Int(cmCMakePresetsErrors::INVALID_VERSION);
2022-03-29 21:10:50 +02:00
2022-08-04 22:12:04 +02:00
auto const VersionHelper = JSONHelperBuilder::Required<int>(
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::NO_VERSION, VersionIntHelper);
2022-03-29 21:10:50 +02:00
2024-11-11 15:18:55 +01:00
auto const VersionRangeHelper = JSONHelperBuilder::Checked<int>(
cmCMakePresetsErrors::UNRECOGNIZED_VERSION_RANGE(MIN_VERSION, MAX_VERSION),
VersionHelper,
[](const int v) -> bool { return v >= MIN_VERSION && v <= MAX_VERSION; });
2022-03-29 21:10:50 +02:00
auto const RootVersionHelper =
2023-12-07 09:12:54 +01:00
JSONHelperBuilder::Object<int>(cmCMakePresetsErrors::INVALID_ROOT_OBJECT)
2024-11-11 15:18:55 +01:00
.Bind("version"_s, VersionRangeHelper, false);
2022-03-29 21:10:50 +02:00
2023-07-02 19:51:09 +02:00
auto const CMakeVersionUIntHelper =
2023-12-07 09:12:54 +01:00
JSONHelperBuilder::UInt(cmCMakePresetsErrors::INVALID_VERSION);
2022-03-29 21:10:50 +02:00
auto const CMakeVersionHelper =
2023-07-02 19:51:09 +02:00
JSONHelperBuilder::Object<CMakeVersion>(JsonErrors::INVALID_NAMED_OBJECT_KEY,
false)
2022-03-29 21:10:50 +02:00
.Bind("major"_s, &CMakeVersion::Major, CMakeVersionUIntHelper, false)
.Bind("minor"_s, &CMakeVersion::Minor, CMakeVersionUIntHelper, false)
.Bind("patch"_s, &CMakeVersion::Patch, CMakeVersionUIntHelper, false);
2023-07-02 19:51:09 +02:00
auto const IncludeHelper =
2023-12-07 09:12:54 +01:00
JSONHelperBuilder::String(cmCMakePresetsErrors::INVALID_INCLUDE);
2022-03-29 21:10:50 +02:00
2022-08-04 22:12:04 +02:00
auto const IncludeVectorHelper = JSONHelperBuilder::Vector<std::string>(
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_INCLUDE, IncludeHelper);
2022-03-29 21:10:50 +02:00
auto const RootPresetsHelper =
2023-07-02 19:51:09 +02:00
JSONHelperBuilder::Object<RootPresets>(
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_ROOT_OBJECT, false)
2022-03-29 21:10:50 +02:00
.Bind<int>("version"_s, nullptr, VersionHelper)
.Bind("configurePresets"_s, &RootPresets::ConfigurePresets,
cmCMakePresetsGraphInternal::ConfigurePresetsHelper, false)
.Bind("buildPresets"_s, &RootPresets::BuildPresets,
cmCMakePresetsGraphInternal::BuildPresetsHelper, false)
.Bind("testPresets"_s, &RootPresets::TestPresets,
cmCMakePresetsGraphInternal::TestPresetsHelper, false)
2022-11-16 20:14:03 +01:00
.Bind("packagePresets"_s, &RootPresets::PackagePresets,
cmCMakePresetsGraphInternal::PackagePresetsHelper, false)
.Bind("workflowPresets"_s, &RootPresets::WorkflowPresets,
cmCMakePresetsGraphInternal::WorkflowPresetsHelper, false)
2022-03-29 21:10:50 +02:00
.Bind("cmakeMinimumRequired"_s, &RootPresets::CMakeMinimumRequired,
CMakeVersionHelper, false)
.Bind("include"_s, &RootPresets::Include, IncludeVectorHelper, false)
2023-07-02 19:51:09 +02:00
.Bind<std::nullptr_t>("vendor"_s, nullptr,
cmCMakePresetsGraphInternal::VendorHelper(
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_ROOT),
false)
.Bind<std::nullptr_t>("$schema"_s, nullptr,
cmCMakePresetsGraphInternal::SchemaHelper(), false);
2024-07-09 14:46:46 +02:00
class EnvironmentMacroExpander : public MacroExpander
{
public:
ExpandMacroResult operator()(const std::string& macroNamespace,
const std::string& macroName,
std::string& macroOut,
int /*version*/) const override
{
if (macroNamespace == "penv") {
if (macroName.empty()) {
return ExpandMacroResult::Error;
}
if (cm::optional<std::string> value =
cmSystemTools::GetEnvVar(macroName)) {
macroOut += *value;
}
return ExpandMacroResult::Ok;
}
return ExpandMacroResult::Ignore;
}
};
2022-03-29 21:10:50 +02:00
}
namespace cmCMakePresetsGraphInternal {
2023-07-02 19:51:09 +02:00
bool PresetStringHelper(std::string& out, const Json::Value* value,
cmJSONState* state)
2022-03-29 21:10:50 +02:00
{
2023-07-02 19:51:09 +02:00
static auto const helper = JSONHelperBuilder::String();
return helper(out, value, state);
}
2022-03-29 21:10:50 +02:00
2023-07-02 19:51:09 +02:00
bool PresetNameHelper(std::string& out, const Json::Value* value,
cmJSONState* state)
{
if (!value || !value->isString() || value->asString().empty()) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_PRESET_NAME(value, state);
2023-07-02 19:51:09 +02:00
return false;
}
out = value->asString();
return true;
2022-03-29 21:10:50 +02:00
}
2023-07-02 19:51:09 +02:00
bool PresetVectorStringHelper(std::vector<std::string>& out,
const Json::Value* value, cmJSONState* state)
2022-03-29 21:10:50 +02:00
{
2022-08-04 22:12:04 +02:00
static auto const helper = JSONHelperBuilder::Vector<std::string>(
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_PRESET,
2022-03-29 21:10:50 +02:00
cmCMakePresetsGraphInternal::PresetStringHelper);
2023-07-02 19:51:09 +02:00
return helper(out, value, state);
2022-03-29 21:10:50 +02:00
}
2023-07-02 19:51:09 +02:00
bool PresetBoolHelper(bool& out, const Json::Value* value, cmJSONState* state)
2022-03-29 21:10:50 +02:00
{
2023-07-02 19:51:09 +02:00
static auto const helper = JSONHelperBuilder::Bool();
return helper(out, value, state);
2022-03-29 21:10:50 +02:00
}
2023-07-02 19:51:09 +02:00
bool PresetOptionalBoolHelper(cm::optional<bool>& out,
const Json::Value* value, cmJSONState* state)
2022-03-29 21:10:50 +02:00
{
2023-07-02 19:51:09 +02:00
static auto const helper =
JSONHelperBuilder::Optional<bool>(PresetBoolHelper);
return helper(out, value, state);
2022-03-29 21:10:50 +02:00
}
2023-07-02 19:51:09 +02:00
bool PresetIntHelper(int& out, const Json::Value* value, cmJSONState* state)
2022-03-29 21:10:50 +02:00
{
2023-07-02 19:51:09 +02:00
static auto const helper = JSONHelperBuilder::Int();
return helper(out, value, state);
2022-03-29 21:10:50 +02:00
}
2023-07-02 19:51:09 +02:00
bool PresetOptionalIntHelper(cm::optional<int>& out, const Json::Value* value,
cmJSONState* state)
2022-03-29 21:10:50 +02:00
{
2023-07-02 19:51:09 +02:00
static auto const helper = JSONHelperBuilder::Optional<int>(PresetIntHelper);
return helper(out, value, state);
2022-03-29 21:10:50 +02:00
}
2023-07-02 19:51:09 +02:00
bool PresetVectorIntHelper(std::vector<int>& out, const Json::Value* value,
cmJSONState* state)
2022-03-29 21:10:50 +02:00
{
2022-08-04 22:12:04 +02:00
static auto const helper = JSONHelperBuilder::Vector<int>(
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_PRESET, PresetIntHelper);
2023-07-02 19:51:09 +02:00
return helper(out, value, state);
2022-03-29 21:10:50 +02:00
}
2023-07-02 19:51:09 +02:00
cmJSONHelper<std::nullptr_t> VendorHelper(const ErrorGenerator& error)
2022-03-29 21:10:50 +02:00
{
2023-07-02 19:51:09 +02:00
return [error](std::nullptr_t& /*out*/, const Json::Value* value,
cmJSONState* state) -> bool {
2022-03-29 21:10:50 +02:00
if (!value) {
2023-07-02 19:51:09 +02:00
return true;
2022-03-29 21:10:50 +02:00
}
if (!value->isObject()) {
2023-07-02 19:51:09 +02:00
error(value, state);
return false;
2022-03-29 21:10:50 +02:00
}
2023-07-02 19:51:09 +02:00
return true;
2022-03-29 21:10:50 +02:00
};
}
2023-07-02 19:51:09 +02:00
bool PresetConditionHelper(
2022-03-29 21:10:50 +02:00
std::shared_ptr<cmCMakePresetsGraph::Condition>& out,
2023-07-02 19:51:09 +02:00
const Json::Value* value, cmJSONState* state)
2022-03-29 21:10:50 +02:00
{
std::unique_ptr<cmCMakePresetsGraph::Condition> ptr;
2023-07-02 19:51:09 +02:00
auto result = ConditionHelper(ptr, value, state);
2022-03-29 21:10:50 +02:00
out = std::move(ptr);
return result;
}
2023-07-02 19:51:09 +02:00
bool PresetVectorOneOrMoreStringHelper(std::vector<std::string>& out,
const Json::Value* value,
cmJSONState* state)
2022-03-29 21:10:50 +02:00
{
out.clear();
if (!value) {
2023-07-02 19:51:09 +02:00
return true;
2022-03-29 21:10:50 +02:00
}
if (value->isString()) {
out.push_back(value->asString());
2023-07-02 19:51:09 +02:00
return true;
2022-03-29 21:10:50 +02:00
}
2023-07-02 19:51:09 +02:00
return PresetVectorStringHelper(out, value, state);
2022-03-29 21:10:50 +02:00
}
2023-07-02 19:51:09 +02:00
bool EnvironmentMapHelper(
2022-03-29 21:10:50 +02:00
std::map<std::string, cm::optional<std::string>>& out,
2023-07-02 19:51:09 +02:00
const Json::Value* value, cmJSONState* state)
2022-03-29 21:10:50 +02:00
{
2022-08-04 22:12:04 +02:00
static auto const helper = JSONHelperBuilder::Map<cm::optional<std::string>>(
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_PRESET, EnvironmentHelper);
2022-03-29 21:10:50 +02:00
2023-07-02 19:51:09 +02:00
return helper(out, value, state);
2022-03-29 21:10:50 +02:00
}
2023-12-07 09:12:54 +01:00
cmJSONHelper<std::nullptr_t> SchemaHelper()
{
return [](std::nullptr_t&, const Json::Value*, cmJSONState*) -> bool {
return true;
};
}
2022-03-29 21:10:50 +02:00
}
2023-07-02 19:51:09 +02:00
bool cmCMakePresetsGraph::ReadJSONFile(const std::string& filename,
RootType rootType,
ReadReason readReason,
std::vector<File*>& inProgressFiles,
File*& file, std::string& errMsg)
2022-03-29 21:10:50 +02:00
{
2023-07-02 19:51:09 +02:00
bool result;
2022-03-29 21:10:50 +02:00
for (auto const& f : this->Files) {
if (cmSystemTools::SameFile(filename, f->Filename)) {
file = f.get();
auto fileIt =
std::find(inProgressFiles.begin(), inProgressFiles.end(), file);
if (fileIt != inProgressFiles.end()) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::CYCLIC_INCLUDE(filename, &this->parseState);
2023-07-02 19:51:09 +02:00
return false;
2022-03-29 21:10:50 +02:00
}
2023-07-02 19:51:09 +02:00
return true;
2022-03-29 21:10:50 +02:00
}
}
Json::Value root;
2023-07-02 19:51:09 +02:00
this->parseState = cmJSONState(filename, &root);
if (!this->parseState.errors.empty()) {
return false;
2022-03-29 21:10:50 +02:00
}
int v = 0;
2023-07-02 19:51:09 +02:00
if ((result = RootVersionHelper(v, &root, &parseState)) != true) {
2022-03-29 21:10:50 +02:00
return result;
}
// Support for build and test presets added in version 2.
2023-07-02 19:51:09 +02:00
if (v < 2) {
if (root.isMember("buildPresets")) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::BUILD_TEST_PRESETS_UNSUPPORTED(
2023-07-02 19:51:09 +02:00
&root["buildPresets"], &this->parseState);
return false;
}
if (root.isMember("testPresets")) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::BUILD_TEST_PRESETS_UNSUPPORTED(
&root["testPresets"], &this->parseState);
2023-07-02 19:51:09 +02:00
return false;
}
2022-03-29 21:10:50 +02:00
}
2022-11-16 20:14:03 +01:00
// Support for package presets added in version 6.
if (v < 6 && root.isMember("packagePresets")) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::PACKAGE_PRESETS_UNSUPPORTED(&root["packagePresets"],
&this->parseState);
2023-07-02 19:51:09 +02:00
return false;
2022-11-16 20:14:03 +01:00
}
// Support for workflow presets added in version 6.
if (v < 6 && root.isMember("workflowPresets")) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::WORKFLOW_PRESETS_UNSUPPORTED(
&root["workflowPresets"], &this->parseState);
2023-07-02 19:51:09 +02:00
return false;
2022-11-16 20:14:03 +01:00
}
2022-03-29 21:10:50 +02:00
// Support for include added in version 4.
if (v < 4 && root.isMember("include")) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INCLUDE_UNSUPPORTED(&root["include"],
&this->parseState);
return false;
}
// Support for $schema added in version 8.
if (v < 8 && root.isMember("$schema")) {
cmCMakePresetsErrors::SCHEMA_UNSUPPORTED(&this->parseState);
2023-07-02 19:51:09 +02:00
return false;
2022-03-29 21:10:50 +02:00
}
2024-11-11 15:18:55 +01:00
// Support for $comment added in version 10.
this->parseState.allowComments = (v >= 10);
2022-03-29 21:10:50 +02:00
RootPresets presets;
2023-07-02 19:51:09 +02:00
if ((result = RootPresetsHelper(presets, &root, &parseState)) != true) {
2022-03-29 21:10:50 +02:00
return result;
}
unsigned int currentMajor = cmVersion::GetMajorVersion();
unsigned int currentMinor = cmVersion::GetMinorVersion();
unsigned int currentPatch = cmVersion::GetPatchVersion();
auto const& required = presets.CMakeMinimumRequired;
2023-07-02 19:51:09 +02:00
if (required.Major > currentMajor) {
2023-12-07 09:12:54 +01:00
ErrorGenerator error = cmCMakePresetsErrors::UNRECOGNIZED_CMAKE_VERSION(
2023-07-02 19:51:09 +02:00
"major", currentMajor, required.Major);
error(&root["cmakeMinimumRequired"]["major"], &this->parseState);
return false;
}
if (required.Major == currentMajor) {
if (required.Minor > currentMinor) {
2023-12-07 09:12:54 +01:00
ErrorGenerator error = cmCMakePresetsErrors::UNRECOGNIZED_CMAKE_VERSION(
2023-07-02 19:51:09 +02:00
"minor", currentMinor, required.Minor);
error(&root["cmakeMinimumRequired"]["minor"], &this->parseState);
return false;
}
if (required.Minor == currentMinor && required.Patch > currentPatch) {
2023-12-07 09:12:54 +01:00
ErrorGenerator error = cmCMakePresetsErrors::UNRECOGNIZED_CMAKE_VERSION(
2023-07-02 19:51:09 +02:00
"patch", currentPatch, required.Patch);
error(&root["cmakeMinimumRequired"]["patch"], &this->parseState);
return false;
}
2022-03-29 21:10:50 +02:00
}
auto filePtr = cm::make_unique<File>();
file = filePtr.get();
this->Files.emplace_back(std::move(filePtr));
inProgressFiles.emplace_back(file);
file->Filename = filename;
file->Version = v;
file->ReachableFiles.insert(file);
for (auto& preset : presets.ConfigurePresets) {
preset.OriginFile = file;
if (preset.Name.empty()) {
2023-07-02 19:51:09 +02:00
// No error, already handled by PresetNameHelper
return false;
2022-03-29 21:10:50 +02:00
}
PresetPair<ConfigurePreset> presetPair;
presetPair.Unexpanded = preset;
presetPair.Expanded = cm::nullopt;
2023-05-23 16:38:00 +02:00
if (!this->ConfigurePresets.emplace(preset.Name, presetPair).second) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::DUPLICATE_PRESETS(preset.Name, &this->parseState);
2023-07-02 19:51:09 +02:00
return false;
2022-03-29 21:10:50 +02:00
}
// Support for installDir presets added in version 3.
if (v < 3 && !preset.InstallDir.empty()) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INSTALL_PREFIX_UNSUPPORTED(&root["installDir"],
&this->parseState);
2023-07-02 19:51:09 +02:00
return false;
2022-03-29 21:10:50 +02:00
}
// Support for conditions added in version 3.
if (v < 3 && preset.ConditionEvaluator) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::CONDITION_UNSUPPORTED(&this->parseState);
2023-07-02 19:51:09 +02:00
return false;
2022-03-29 21:10:50 +02:00
}
// Support for toolchainFile presets added in version 3.
if (v < 3 && !preset.ToolchainFile.empty()) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::TOOLCHAIN_FILE_UNSUPPORTED(&this->parseState);
2023-07-02 19:51:09 +02:00
return false;
}
// Support for trace presets added in version 7.
if (v < 7 &&
(preset.TraceMode.has_value() || preset.TraceFormat.has_value() ||
!preset.TraceRedirect.empty() || !preset.TraceSource.empty())) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::TRACE_UNSUPPORTED(&this->parseState);
2023-07-02 19:51:09 +02:00
return false;
2022-03-29 21:10:50 +02:00
}
2024-11-11 15:18:55 +01:00
// Support for graphviz argument added in version 10.
if (v < 10 && !preset.GraphVizFile.empty()) {
cmCMakePresetsErrors::GRAPHVIZ_FILE_UNSUPPORTED(&this->parseState);
return false;
}
2022-03-29 21:10:50 +02:00
this->ConfigurePresetOrder.push_back(preset.Name);
}
for (auto& preset : presets.BuildPresets) {
preset.OriginFile = file;
if (preset.Name.empty()) {
2023-07-02 19:51:09 +02:00
// No error, already handled by PresetNameHelper
return false;
2022-03-29 21:10:50 +02:00
}
PresetPair<BuildPreset> presetPair;
presetPair.Unexpanded = preset;
presetPair.Expanded = cm::nullopt;
if (!this->BuildPresets.emplace(preset.Name, presetPair).second) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::DUPLICATE_PRESETS(preset.Name, &this->parseState);
2023-07-02 19:51:09 +02:00
return false;
2022-03-29 21:10:50 +02:00
}
// Support for conditions added in version 3.
if (v < 3 && preset.ConditionEvaluator) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::CONDITION_UNSUPPORTED(&this->parseState);
2023-07-02 19:51:09 +02:00
return false;
2022-03-29 21:10:50 +02:00
}
this->BuildPresetOrder.push_back(preset.Name);
}
for (auto& preset : presets.TestPresets) {
preset.OriginFile = file;
if (preset.Name.empty()) {
2023-07-02 19:51:09 +02:00
// No error, already handled by PresetNameHelper
return false;
2022-03-29 21:10:50 +02:00
}
PresetPair<TestPreset> presetPair;
presetPair.Unexpanded = preset;
presetPair.Expanded = cm::nullopt;
if (!this->TestPresets.emplace(preset.Name, presetPair).second) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::DUPLICATE_PRESETS(preset.Name, &this->parseState);
2023-07-02 19:51:09 +02:00
return false;
2022-03-29 21:10:50 +02:00
}
// Support for conditions added in version 3.
if (v < 3 && preset.ConditionEvaluator) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::CONDITION_UNSUPPORTED(&this->parseState);
2023-07-02 19:51:09 +02:00
return false;
2022-03-29 21:10:50 +02:00
}
2022-08-04 22:12:04 +02:00
// Support for TestOutputTruncation added in version 5.
if (v < 5 && preset.Output && preset.Output->TestOutputTruncation) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::TEST_OUTPUT_TRUNCATION_UNSUPPORTED(
2023-07-02 19:51:09 +02:00
&this->parseState);
return false;
2022-08-04 22:12:04 +02:00
}
2022-11-16 20:14:03 +01:00
// Support for outputJUnitFile added in version 6.
if (v < 6 && preset.Output && !preset.Output->OutputJUnitFile.empty()) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::CTEST_JUNIT_UNSUPPORTED(&this->parseState);
2023-07-02 19:51:09 +02:00
return false;
2022-11-16 20:14:03 +01:00
}
2022-03-29 21:10:50 +02:00
this->TestPresetOrder.push_back(preset.Name);
}
2022-11-16 20:14:03 +01:00
for (auto& preset : presets.PackagePresets) {
preset.OriginFile = file;
if (preset.Name.empty()) {
2023-07-02 19:51:09 +02:00
// No error, already handled by PresetNameHelper
return false;
2022-11-16 20:14:03 +01:00
}
PresetPair<PackagePreset> presetPair;
presetPair.Unexpanded = preset;
presetPair.Expanded = cm::nullopt;
if (!this->PackagePresets.emplace(preset.Name, presetPair).second) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::DUPLICATE_PRESETS(preset.Name, &this->parseState);
2023-07-02 19:51:09 +02:00
return false;
2022-11-16 20:14:03 +01:00
}
// Support for conditions added in version 3, but this requires version 5
// already, so no action needed.
this->PackagePresetOrder.push_back(preset.Name);
}
for (auto& preset : presets.WorkflowPresets) {
preset.OriginFile = file;
if (preset.Name.empty()) {
2023-07-02 19:51:09 +02:00
// No error, already handled by PresetNameHelper
return false;
2022-11-16 20:14:03 +01:00
}
PresetPair<WorkflowPreset> presetPair;
presetPair.Unexpanded = preset;
presetPair.Expanded = cm::nullopt;
if (!this->WorkflowPresets.emplace(preset.Name, presetPair).second) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::DUPLICATE_PRESETS(preset.Name, &this->parseState);
2023-07-02 19:51:09 +02:00
return false;
2022-11-16 20:14:03 +01:00
}
// Support for conditions added in version 3, but this requires version 6
// already, so no action needed.
this->WorkflowPresetOrder.push_back(preset.Name);
}
2023-07-02 19:51:09 +02:00
auto const includeFile = [this, &inProgressFiles,
file](const std::string& include,
RootType rootType2, ReadReason readReason2,
std::string& FailureMessage) -> bool {
bool r;
2022-03-29 21:10:50 +02:00
File* includedFile;
2023-07-02 19:51:09 +02:00
if ((r =
this->ReadJSONFile(include, rootType2, readReason2, inProgressFiles,
includedFile, FailureMessage)) != true) {
2022-03-29 21:10:50 +02:00
return r;
}
file->ReachableFiles.insert(includedFile->ReachableFiles.begin(),
includedFile->ReachableFiles.end());
2023-07-02 19:51:09 +02:00
return true;
};
2024-07-09 14:46:46 +02:00
MacroExpanderVector macroExpanders{};
2023-07-02 19:51:09 +02:00
2024-07-09 14:46:46 +02:00
if (v >= 9) {
macroExpanders.push_back(
cm::make_unique<BaseMacroExpander>(*this, filename));
}
macroExpanders.push_back(cm::make_unique<EnvironmentMacroExpander>());
2023-07-02 19:51:09 +02:00
for (Json::ArrayIndex i = 0; i < presets.Include.size(); ++i) {
auto include = presets.Include[i];
// Support for macro expansion in includes added in version 7
if (v >= 7) {
if (ExpandMacros(include, macroExpanders, v) != ExpandMacroResult::Ok) {
2023-12-07 09:12:54 +01:00
cmCMakePresetsErrors::INVALID_INCLUDE(&root["include"][i],
&this->parseState);
2023-07-02 19:51:09 +02:00
return false;
}
}
2022-03-29 21:10:50 +02:00
if (!cmSystemTools::FileIsFullPath(include)) {
auto directory = cmSystemTools::GetFilenamePath(filename);
include = cmStrCat(directory, '/', include);
}
2022-11-16 20:14:03 +01:00
if ((result = includeFile(include, rootType, ReadReason::Included,
2023-07-02 19:51:09 +02:00
errMsg)) != true) {
2022-03-29 21:10:50 +02:00
return result;
}
}
if (rootType == RootType::User && readReason == ReadReason::Root) {
auto cmakePresetsFilename = GetFilename(this->SourceDir);
if (cmSystemTools::FileExists(cmakePresetsFilename)) {
if ((result = includeFile(cmakePresetsFilename, RootType::Project,
2023-07-02 19:51:09 +02:00
ReadReason::Root, errMsg)) != true) {
2022-03-29 21:10:50 +02:00
return result;
}
}
}
inProgressFiles.pop_back();
2023-07-02 19:51:09 +02:00
return true;
2022-03-29 21:10:50 +02:00
}