cmCMakePresetsGraphReadJSON.cxx 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765
  1. /* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
  2. file Copyright.txt or https://cmake.org/licensing for details. */
  3. #include <algorithm>
  4. #include <fstream>
  5. #include <functional>
  6. #include <map>
  7. #include <string>
  8. #include <unordered_set>
  9. #include <utility>
  10. #include <vector>
  11. #include <cm/memory>
  12. #include <cm/optional>
  13. #include <cmext/string_view>
  14. #include <cm3p/json/value.h>
  15. #include "cmCMakePresetsErrors.h"
  16. #include "cmCMakePresetsGraph.h"
  17. #include "cmCMakePresetsGraphInternal.h"
  18. #include "cmJSONHelpers.h"
  19. #include "cmJSONState.h"
  20. #include "cmStringAlgorithms.h"
  21. #include "cmSystemTools.h"
  22. #include "cmVersion.h"
  23. namespace {
  24. using CacheVariable = cmCMakePresetsGraph::CacheVariable;
  25. using ConfigurePreset = cmCMakePresetsGraph::ConfigurePreset;
  26. using BuildPreset = cmCMakePresetsGraph::BuildPreset;
  27. using TestPreset = cmCMakePresetsGraph::TestPreset;
  28. using PackagePreset = cmCMakePresetsGraph::PackagePreset;
  29. using WorkflowPreset = cmCMakePresetsGraph::WorkflowPreset;
  30. using ArchToolsetStrategy = cmCMakePresetsGraph::ArchToolsetStrategy;
  31. using JSONHelperBuilder = cmJSONHelperBuilder;
  32. using ExpandMacroResult = cmCMakePresetsGraphInternal::ExpandMacroResult;
  33. using MacroExpander = cmCMakePresetsGraphInternal::MacroExpander;
  34. using cmCMakePresetsGraphInternal::ExpandMacros;
  35. constexpr int MIN_VERSION = 1;
  36. constexpr int MAX_VERSION = 8;
  37. struct CMakeVersion
  38. {
  39. unsigned int Major = 0;
  40. unsigned int Minor = 0;
  41. unsigned int Patch = 0;
  42. };
  43. struct RootPresets
  44. {
  45. CMakeVersion CMakeMinimumRequired;
  46. std::vector<ConfigurePreset> ConfigurePresets;
  47. std::vector<BuildPreset> BuildPresets;
  48. std::vector<TestPreset> TestPresets;
  49. std::vector<PackagePreset> PackagePresets;
  50. std::vector<WorkflowPreset> WorkflowPresets;
  51. std::vector<std::string> Include;
  52. };
  53. std::unique_ptr<cmCMakePresetsGraphInternal::NotCondition> InvertCondition(
  54. std::unique_ptr<cmCMakePresetsGraph::Condition> condition)
  55. {
  56. auto retval = cm::make_unique<cmCMakePresetsGraphInternal::NotCondition>();
  57. retval->SubCondition = std::move(condition);
  58. return retval;
  59. }
  60. auto const ConditionStringHelper = JSONHelperBuilder::String();
  61. auto const ConditionBoolHelper = JSONHelperBuilder::Bool();
  62. auto const ConditionStringListHelper = JSONHelperBuilder::Vector<std::string>(
  63. cmCMakePresetsErrors::INVALID_CONDITION, ConditionStringHelper);
  64. auto const ConstConditionHelper =
  65. JSONHelperBuilder::Object<cmCMakePresetsGraphInternal::ConstCondition>(
  66. cmCMakePresetsErrors::INVALID_CONDITION_OBJECT, false)
  67. .Bind<std::string>("type"_s, nullptr, ConditionStringHelper, true)
  68. .Bind("value"_s, &cmCMakePresetsGraphInternal::ConstCondition::Value,
  69. ConditionBoolHelper, true);
  70. auto const EqualsConditionHelper =
  71. JSONHelperBuilder::Object<cmCMakePresetsGraphInternal::EqualsCondition>(
  72. cmCMakePresetsErrors::INVALID_CONDITION_OBJECT, false)
  73. .Bind<std::string>("type"_s, nullptr, ConditionStringHelper, true)
  74. .Bind("lhs"_s, &cmCMakePresetsGraphInternal::EqualsCondition::Lhs,
  75. ConditionStringHelper, true)
  76. .Bind("rhs"_s, &cmCMakePresetsGraphInternal::EqualsCondition::Rhs,
  77. ConditionStringHelper, true);
  78. auto const InListConditionHelper =
  79. JSONHelperBuilder::Object<cmCMakePresetsGraphInternal::InListCondition>(
  80. cmCMakePresetsErrors::INVALID_CONDITION_OBJECT, false)
  81. .Bind<std::string>("type"_s, nullptr, ConditionStringHelper, true)
  82. .Bind("string"_s, &cmCMakePresetsGraphInternal::InListCondition::String,
  83. ConditionStringHelper, true)
  84. .Bind("list"_s, &cmCMakePresetsGraphInternal::InListCondition::List,
  85. ConditionStringListHelper, true);
  86. auto const MatchesConditionHelper =
  87. JSONHelperBuilder::Object<cmCMakePresetsGraphInternal::MatchesCondition>(
  88. cmCMakePresetsErrors::INVALID_CONDITION_OBJECT, false)
  89. .Bind<std::string>("type"_s, nullptr, ConditionStringHelper, true)
  90. .Bind("string"_s, &cmCMakePresetsGraphInternal::MatchesCondition::String,
  91. ConditionStringHelper, true)
  92. .Bind("regex"_s, &cmCMakePresetsGraphInternal::MatchesCondition::Regex,
  93. ConditionStringHelper, true);
  94. bool SubConditionHelper(std::unique_ptr<cmCMakePresetsGraph::Condition>& out,
  95. const Json::Value* value, cmJSONState* state);
  96. auto const ListConditionVectorHelper =
  97. JSONHelperBuilder::Vector<std::unique_ptr<cmCMakePresetsGraph::Condition>>(
  98. cmCMakePresetsErrors::INVALID_CONDITION, SubConditionHelper);
  99. auto const AnyAllOfConditionHelper =
  100. JSONHelperBuilder::Object<cmCMakePresetsGraphInternal::AnyAllOfCondition>(
  101. cmCMakePresetsErrors::INVALID_CONDITION_OBJECT, false)
  102. .Bind<std::string>("type"_s, nullptr, ConditionStringHelper, true)
  103. .Bind("conditions"_s,
  104. &cmCMakePresetsGraphInternal::AnyAllOfCondition::Conditions,
  105. ListConditionVectorHelper);
  106. auto const NotConditionHelper =
  107. JSONHelperBuilder::Object<cmCMakePresetsGraphInternal::NotCondition>(
  108. cmCMakePresetsErrors::INVALID_CONDITION_OBJECT, false)
  109. .Bind<std::string>("type"_s, nullptr, ConditionStringHelper, true)
  110. .Bind("condition"_s,
  111. &cmCMakePresetsGraphInternal::NotCondition::SubCondition,
  112. SubConditionHelper);
  113. bool ConditionHelper(std::unique_ptr<cmCMakePresetsGraph::Condition>& out,
  114. const Json::Value* value, cmJSONState* state)
  115. {
  116. if (!value) {
  117. out.reset();
  118. return true;
  119. }
  120. if (value->isBool()) {
  121. auto c = cm::make_unique<cmCMakePresetsGraphInternal::ConstCondition>();
  122. c->Value = value->asBool();
  123. out = std::move(c);
  124. return true;
  125. }
  126. if (value->isNull()) {
  127. out = cm::make_unique<cmCMakePresetsGraphInternal::NullCondition>();
  128. return true;
  129. }
  130. if (value->isObject()) {
  131. if (!value->isMember("type")) {
  132. cmCMakePresetsErrors::INVALID_CONDITION(value, state);
  133. return false;
  134. }
  135. if (!(*value)["type"].isString()) {
  136. cmCMakePresetsErrors::INVALID_CONDITION(value, state);
  137. return false;
  138. }
  139. auto type = (*value)["type"].asString();
  140. if (type == "const") {
  141. auto c = cm::make_unique<cmCMakePresetsGraphInternal::ConstCondition>();
  142. CHECK_OK(ConstConditionHelper(*c, value, state));
  143. out = std::move(c);
  144. return true;
  145. }
  146. if (type == "equals" || type == "notEquals") {
  147. auto c = cm::make_unique<cmCMakePresetsGraphInternal::EqualsCondition>();
  148. CHECK_OK(EqualsConditionHelper(*c, value, state));
  149. out = std::move(c);
  150. if (type == "notEquals") {
  151. out = InvertCondition(std::move(out));
  152. }
  153. return true;
  154. }
  155. if (type == "inList" || type == "notInList") {
  156. auto c = cm::make_unique<cmCMakePresetsGraphInternal::InListCondition>();
  157. CHECK_OK(InListConditionHelper(*c, value, state));
  158. out = std::move(c);
  159. if (type == "notInList") {
  160. out = InvertCondition(std::move(out));
  161. }
  162. return true;
  163. }
  164. if (type == "matches" || type == "notMatches") {
  165. auto c =
  166. cm::make_unique<cmCMakePresetsGraphInternal::MatchesCondition>();
  167. CHECK_OK(MatchesConditionHelper(*c, value, state));
  168. out = std::move(c);
  169. if (type == "notMatches") {
  170. out = InvertCondition(std::move(out));
  171. }
  172. return true;
  173. }
  174. if (type == "anyOf" || type == "allOf") {
  175. auto c =
  176. cm::make_unique<cmCMakePresetsGraphInternal::AnyAllOfCondition>();
  177. c->StopValue = (type == "anyOf");
  178. CHECK_OK(AnyAllOfConditionHelper(*c, value, state));
  179. out = std::move(c);
  180. return true;
  181. }
  182. if (type == "not") {
  183. auto c = cm::make_unique<cmCMakePresetsGraphInternal::NotCondition>();
  184. CHECK_OK(NotConditionHelper(*c, value, state));
  185. out = std::move(c);
  186. return true;
  187. }
  188. }
  189. cmCMakePresetsErrors::INVALID_CONDITION(value, state);
  190. return false;
  191. }
  192. bool SubConditionHelper(std::unique_ptr<cmCMakePresetsGraph::Condition>& out,
  193. const Json::Value* value, cmJSONState* state)
  194. {
  195. std::unique_ptr<cmCMakePresetsGraph::Condition> ptr;
  196. auto result = ConditionHelper(ptr, value, state);
  197. if (ptr && ptr->IsNull()) {
  198. cmCMakePresetsErrors::INVALID_CONDITION(value, state);
  199. return false;
  200. }
  201. out = std::move(ptr);
  202. return result;
  203. }
  204. bool EnvironmentHelper(cm::optional<std::string>& out,
  205. const Json::Value* value, cmJSONState* state)
  206. {
  207. if (!value || value->isNull()) {
  208. out = cm::nullopt;
  209. return true;
  210. }
  211. if (value->isString()) {
  212. out = value->asString();
  213. return true;
  214. }
  215. cmCMakePresetsErrors::INVALID_PRESET(value, state);
  216. return false;
  217. }
  218. auto const VersionIntHelper =
  219. JSONHelperBuilder::Int(cmCMakePresetsErrors::INVALID_VERSION);
  220. auto const VersionHelper = JSONHelperBuilder::Required<int>(
  221. cmCMakePresetsErrors::NO_VERSION, VersionIntHelper);
  222. auto const RootVersionHelper =
  223. JSONHelperBuilder::Object<int>(cmCMakePresetsErrors::INVALID_ROOT_OBJECT)
  224. .Bind("version"_s, VersionHelper, false);
  225. auto const CMakeVersionUIntHelper =
  226. JSONHelperBuilder::UInt(cmCMakePresetsErrors::INVALID_VERSION);
  227. auto const CMakeVersionHelper =
  228. JSONHelperBuilder::Object<CMakeVersion>(JsonErrors::INVALID_NAMED_OBJECT_KEY,
  229. false)
  230. .Bind("major"_s, &CMakeVersion::Major, CMakeVersionUIntHelper, false)
  231. .Bind("minor"_s, &CMakeVersion::Minor, CMakeVersionUIntHelper, false)
  232. .Bind("patch"_s, &CMakeVersion::Patch, CMakeVersionUIntHelper, false);
  233. auto const IncludeHelper =
  234. JSONHelperBuilder::String(cmCMakePresetsErrors::INVALID_INCLUDE);
  235. auto const IncludeVectorHelper = JSONHelperBuilder::Vector<std::string>(
  236. cmCMakePresetsErrors::INVALID_INCLUDE, IncludeHelper);
  237. auto const RootPresetsHelper =
  238. JSONHelperBuilder::Object<RootPresets>(
  239. cmCMakePresetsErrors::INVALID_ROOT_OBJECT, false)
  240. .Bind<int>("version"_s, nullptr, VersionHelper)
  241. .Bind("configurePresets"_s, &RootPresets::ConfigurePresets,
  242. cmCMakePresetsGraphInternal::ConfigurePresetsHelper, false)
  243. .Bind("buildPresets"_s, &RootPresets::BuildPresets,
  244. cmCMakePresetsGraphInternal::BuildPresetsHelper, false)
  245. .Bind("testPresets"_s, &RootPresets::TestPresets,
  246. cmCMakePresetsGraphInternal::TestPresetsHelper, false)
  247. .Bind("packagePresets"_s, &RootPresets::PackagePresets,
  248. cmCMakePresetsGraphInternal::PackagePresetsHelper, false)
  249. .Bind("workflowPresets"_s, &RootPresets::WorkflowPresets,
  250. cmCMakePresetsGraphInternal::WorkflowPresetsHelper, false)
  251. .Bind("cmakeMinimumRequired"_s, &RootPresets::CMakeMinimumRequired,
  252. CMakeVersionHelper, false)
  253. .Bind("include"_s, &RootPresets::Include, IncludeVectorHelper, false)
  254. .Bind<std::nullptr_t>("vendor"_s, nullptr,
  255. cmCMakePresetsGraphInternal::VendorHelper(
  256. cmCMakePresetsErrors::INVALID_ROOT),
  257. false)
  258. .Bind<std::nullptr_t>("$schema"_s, nullptr,
  259. cmCMakePresetsGraphInternal::SchemaHelper(), false);
  260. }
  261. namespace cmCMakePresetsGraphInternal {
  262. bool PresetStringHelper(std::string& out, const Json::Value* value,
  263. cmJSONState* state)
  264. {
  265. static auto const helper = JSONHelperBuilder::String();
  266. return helper(out, value, state);
  267. }
  268. bool PresetNameHelper(std::string& out, const Json::Value* value,
  269. cmJSONState* state)
  270. {
  271. if (!value || !value->isString() || value->asString().empty()) {
  272. cmCMakePresetsErrors::INVALID_PRESET_NAME(value, state);
  273. return false;
  274. }
  275. out = value->asString();
  276. return true;
  277. }
  278. bool PresetVectorStringHelper(std::vector<std::string>& out,
  279. const Json::Value* value, cmJSONState* state)
  280. {
  281. static auto const helper = JSONHelperBuilder::Vector<std::string>(
  282. cmCMakePresetsErrors::INVALID_PRESET,
  283. cmCMakePresetsGraphInternal::PresetStringHelper);
  284. return helper(out, value, state);
  285. }
  286. bool PresetBoolHelper(bool& out, const Json::Value* value, cmJSONState* state)
  287. {
  288. static auto const helper = JSONHelperBuilder::Bool();
  289. return helper(out, value, state);
  290. }
  291. bool PresetOptionalBoolHelper(cm::optional<bool>& out,
  292. const Json::Value* value, cmJSONState* state)
  293. {
  294. static auto const helper =
  295. JSONHelperBuilder::Optional<bool>(PresetBoolHelper);
  296. return helper(out, value, state);
  297. }
  298. bool PresetIntHelper(int& out, const Json::Value* value, cmJSONState* state)
  299. {
  300. static auto const helper = JSONHelperBuilder::Int();
  301. return helper(out, value, state);
  302. }
  303. bool PresetOptionalIntHelper(cm::optional<int>& out, const Json::Value* value,
  304. cmJSONState* state)
  305. {
  306. static auto const helper = JSONHelperBuilder::Optional<int>(PresetIntHelper);
  307. return helper(out, value, state);
  308. }
  309. bool PresetVectorIntHelper(std::vector<int>& out, const Json::Value* value,
  310. cmJSONState* state)
  311. {
  312. static auto const helper = JSONHelperBuilder::Vector<int>(
  313. cmCMakePresetsErrors::INVALID_PRESET, PresetIntHelper);
  314. return helper(out, value, state);
  315. }
  316. cmJSONHelper<std::nullptr_t> VendorHelper(const ErrorGenerator& error)
  317. {
  318. return [error](std::nullptr_t& /*out*/, const Json::Value* value,
  319. cmJSONState* state) -> bool {
  320. if (!value) {
  321. return true;
  322. }
  323. if (!value->isObject()) {
  324. error(value, state);
  325. return false;
  326. }
  327. return true;
  328. };
  329. }
  330. bool PresetConditionHelper(
  331. std::shared_ptr<cmCMakePresetsGraph::Condition>& out,
  332. const Json::Value* value, cmJSONState* state)
  333. {
  334. std::unique_ptr<cmCMakePresetsGraph::Condition> ptr;
  335. auto result = ConditionHelper(ptr, value, state);
  336. out = std::move(ptr);
  337. return result;
  338. }
  339. bool PresetVectorOneOrMoreStringHelper(std::vector<std::string>& out,
  340. const Json::Value* value,
  341. cmJSONState* state)
  342. {
  343. out.clear();
  344. if (!value) {
  345. return true;
  346. }
  347. if (value->isString()) {
  348. out.push_back(value->asString());
  349. return true;
  350. }
  351. return PresetVectorStringHelper(out, value, state);
  352. }
  353. bool EnvironmentMapHelper(
  354. std::map<std::string, cm::optional<std::string>>& out,
  355. const Json::Value* value, cmJSONState* state)
  356. {
  357. static auto const helper = JSONHelperBuilder::Map<cm::optional<std::string>>(
  358. cmCMakePresetsErrors::INVALID_PRESET, EnvironmentHelper);
  359. return helper(out, value, state);
  360. }
  361. cmJSONHelper<std::nullptr_t> SchemaHelper()
  362. {
  363. return [](std::nullptr_t&, const Json::Value*, cmJSONState*) -> bool {
  364. return true;
  365. };
  366. }
  367. }
  368. bool cmCMakePresetsGraph::ReadJSONFile(const std::string& filename,
  369. RootType rootType,
  370. ReadReason readReason,
  371. std::vector<File*>& inProgressFiles,
  372. File*& file, std::string& errMsg)
  373. {
  374. bool result;
  375. for (auto const& f : this->Files) {
  376. if (cmSystemTools::SameFile(filename, f->Filename)) {
  377. file = f.get();
  378. auto fileIt =
  379. std::find(inProgressFiles.begin(), inProgressFiles.end(), file);
  380. if (fileIt != inProgressFiles.end()) {
  381. cmCMakePresetsErrors::CYCLIC_INCLUDE(filename, &this->parseState);
  382. return false;
  383. }
  384. return true;
  385. }
  386. }
  387. Json::Value root;
  388. this->parseState = cmJSONState(filename, &root);
  389. if (!this->parseState.errors.empty()) {
  390. return false;
  391. }
  392. int v = 0;
  393. if ((result = RootVersionHelper(v, &root, &parseState)) != true) {
  394. return result;
  395. }
  396. if (v < MIN_VERSION || v > MAX_VERSION) {
  397. cmCMakePresetsErrors::UNRECOGNIZED_VERSION(&root["version"],
  398. &this->parseState);
  399. return false;
  400. }
  401. // Support for build and test presets added in version 2.
  402. if (v < 2) {
  403. if (root.isMember("buildPresets")) {
  404. cmCMakePresetsErrors::BUILD_TEST_PRESETS_UNSUPPORTED(
  405. &root["buildPresets"], &this->parseState);
  406. return false;
  407. }
  408. if (root.isMember("testPresets")) {
  409. cmCMakePresetsErrors::BUILD_TEST_PRESETS_UNSUPPORTED(
  410. &root["testPresets"], &this->parseState);
  411. return false;
  412. }
  413. }
  414. // Support for package presets added in version 6.
  415. if (v < 6 && root.isMember("packagePresets")) {
  416. cmCMakePresetsErrors::PACKAGE_PRESETS_UNSUPPORTED(&root["packagePresets"],
  417. &this->parseState);
  418. return false;
  419. }
  420. // Support for workflow presets added in version 6.
  421. if (v < 6 && root.isMember("workflowPresets")) {
  422. cmCMakePresetsErrors::WORKFLOW_PRESETS_UNSUPPORTED(
  423. &root["workflowPresets"], &this->parseState);
  424. return false;
  425. }
  426. // Support for include added in version 4.
  427. if (v < 4 && root.isMember("include")) {
  428. cmCMakePresetsErrors::INCLUDE_UNSUPPORTED(&root["include"],
  429. &this->parseState);
  430. return false;
  431. }
  432. // Support for $schema added in version 8.
  433. if (v < 8 && root.isMember("$schema")) {
  434. cmCMakePresetsErrors::SCHEMA_UNSUPPORTED(&this->parseState);
  435. return false;
  436. }
  437. RootPresets presets;
  438. if ((result = RootPresetsHelper(presets, &root, &parseState)) != true) {
  439. return result;
  440. }
  441. unsigned int currentMajor = cmVersion::GetMajorVersion();
  442. unsigned int currentMinor = cmVersion::GetMinorVersion();
  443. unsigned int currentPatch = cmVersion::GetPatchVersion();
  444. auto const& required = presets.CMakeMinimumRequired;
  445. if (required.Major > currentMajor) {
  446. ErrorGenerator error = cmCMakePresetsErrors::UNRECOGNIZED_CMAKE_VERSION(
  447. "major", currentMajor, required.Major);
  448. error(&root["cmakeMinimumRequired"]["major"], &this->parseState);
  449. return false;
  450. }
  451. if (required.Major == currentMajor) {
  452. if (required.Minor > currentMinor) {
  453. ErrorGenerator error = cmCMakePresetsErrors::UNRECOGNIZED_CMAKE_VERSION(
  454. "minor", currentMinor, required.Minor);
  455. error(&root["cmakeMinimumRequired"]["minor"], &this->parseState);
  456. return false;
  457. }
  458. if (required.Minor == currentMinor && required.Patch > currentPatch) {
  459. ErrorGenerator error = cmCMakePresetsErrors::UNRECOGNIZED_CMAKE_VERSION(
  460. "patch", currentPatch, required.Patch);
  461. error(&root["cmakeMinimumRequired"]["patch"], &this->parseState);
  462. return false;
  463. }
  464. }
  465. auto filePtr = cm::make_unique<File>();
  466. file = filePtr.get();
  467. this->Files.emplace_back(std::move(filePtr));
  468. inProgressFiles.emplace_back(file);
  469. file->Filename = filename;
  470. file->Version = v;
  471. file->ReachableFiles.insert(file);
  472. for (auto& preset : presets.ConfigurePresets) {
  473. preset.OriginFile = file;
  474. if (preset.Name.empty()) {
  475. // No error, already handled by PresetNameHelper
  476. return false;
  477. }
  478. PresetPair<ConfigurePreset> presetPair;
  479. presetPair.Unexpanded = preset;
  480. presetPair.Expanded = cm::nullopt;
  481. if (!this->ConfigurePresets.emplace(preset.Name, presetPair).second) {
  482. cmCMakePresetsErrors::DUPLICATE_PRESETS(preset.Name, &this->parseState);
  483. return false;
  484. }
  485. // Support for installDir presets added in version 3.
  486. if (v < 3 && !preset.InstallDir.empty()) {
  487. cmCMakePresetsErrors::INSTALL_PREFIX_UNSUPPORTED(&root["installDir"],
  488. &this->parseState);
  489. return false;
  490. }
  491. // Support for conditions added in version 3.
  492. if (v < 3 && preset.ConditionEvaluator) {
  493. cmCMakePresetsErrors::CONDITION_UNSUPPORTED(&this->parseState);
  494. return false;
  495. }
  496. // Support for toolchainFile presets added in version 3.
  497. if (v < 3 && !preset.ToolchainFile.empty()) {
  498. cmCMakePresetsErrors::TOOLCHAIN_FILE_UNSUPPORTED(&this->parseState);
  499. return false;
  500. }
  501. // Support for trace presets added in version 7.
  502. if (v < 7 &&
  503. (preset.TraceMode.has_value() || preset.TraceFormat.has_value() ||
  504. !preset.TraceRedirect.empty() || !preset.TraceSource.empty())) {
  505. cmCMakePresetsErrors::TRACE_UNSUPPORTED(&this->parseState);
  506. return false;
  507. }
  508. this->ConfigurePresetOrder.push_back(preset.Name);
  509. }
  510. for (auto& preset : presets.BuildPresets) {
  511. preset.OriginFile = file;
  512. if (preset.Name.empty()) {
  513. // No error, already handled by PresetNameHelper
  514. return false;
  515. }
  516. PresetPair<BuildPreset> presetPair;
  517. presetPair.Unexpanded = preset;
  518. presetPair.Expanded = cm::nullopt;
  519. if (!this->BuildPresets.emplace(preset.Name, presetPair).second) {
  520. cmCMakePresetsErrors::DUPLICATE_PRESETS(preset.Name, &this->parseState);
  521. return false;
  522. }
  523. // Support for conditions added in version 3.
  524. if (v < 3 && preset.ConditionEvaluator) {
  525. cmCMakePresetsErrors::CONDITION_UNSUPPORTED(&this->parseState);
  526. return false;
  527. }
  528. this->BuildPresetOrder.push_back(preset.Name);
  529. }
  530. for (auto& preset : presets.TestPresets) {
  531. preset.OriginFile = file;
  532. if (preset.Name.empty()) {
  533. // No error, already handled by PresetNameHelper
  534. return false;
  535. }
  536. PresetPair<TestPreset> presetPair;
  537. presetPair.Unexpanded = preset;
  538. presetPair.Expanded = cm::nullopt;
  539. if (!this->TestPresets.emplace(preset.Name, presetPair).second) {
  540. cmCMakePresetsErrors::DUPLICATE_PRESETS(preset.Name, &this->parseState);
  541. return false;
  542. }
  543. // Support for conditions added in version 3.
  544. if (v < 3 && preset.ConditionEvaluator) {
  545. cmCMakePresetsErrors::CONDITION_UNSUPPORTED(&this->parseState);
  546. return false;
  547. }
  548. // Support for TestOutputTruncation added in version 5.
  549. if (v < 5 && preset.Output && preset.Output->TestOutputTruncation) {
  550. cmCMakePresetsErrors::TEST_OUTPUT_TRUNCATION_UNSUPPORTED(
  551. &this->parseState);
  552. return false;
  553. }
  554. // Support for outputJUnitFile added in version 6.
  555. if (v < 6 && preset.Output && !preset.Output->OutputJUnitFile.empty()) {
  556. cmCMakePresetsErrors::CTEST_JUNIT_UNSUPPORTED(&this->parseState);
  557. return false;
  558. }
  559. this->TestPresetOrder.push_back(preset.Name);
  560. }
  561. for (auto& preset : presets.PackagePresets) {
  562. preset.OriginFile = file;
  563. if (preset.Name.empty()) {
  564. // No error, already handled by PresetNameHelper
  565. return false;
  566. }
  567. PresetPair<PackagePreset> presetPair;
  568. presetPair.Unexpanded = preset;
  569. presetPair.Expanded = cm::nullopt;
  570. if (!this->PackagePresets.emplace(preset.Name, presetPair).second) {
  571. cmCMakePresetsErrors::DUPLICATE_PRESETS(preset.Name, &this->parseState);
  572. return false;
  573. }
  574. // Support for conditions added in version 3, but this requires version 5
  575. // already, so no action needed.
  576. this->PackagePresetOrder.push_back(preset.Name);
  577. }
  578. for (auto& preset : presets.WorkflowPresets) {
  579. preset.OriginFile = file;
  580. if (preset.Name.empty()) {
  581. // No error, already handled by PresetNameHelper
  582. return false;
  583. }
  584. PresetPair<WorkflowPreset> presetPair;
  585. presetPair.Unexpanded = preset;
  586. presetPair.Expanded = cm::nullopt;
  587. if (!this->WorkflowPresets.emplace(preset.Name, presetPair).second) {
  588. cmCMakePresetsErrors::DUPLICATE_PRESETS(preset.Name, &this->parseState);
  589. return false;
  590. }
  591. // Support for conditions added in version 3, but this requires version 6
  592. // already, so no action needed.
  593. this->WorkflowPresetOrder.push_back(preset.Name);
  594. }
  595. auto const includeFile = [this, &inProgressFiles,
  596. file](const std::string& include,
  597. RootType rootType2, ReadReason readReason2,
  598. std::string& FailureMessage) -> bool {
  599. bool r;
  600. File* includedFile;
  601. if ((r =
  602. this->ReadJSONFile(include, rootType2, readReason2, inProgressFiles,
  603. includedFile, FailureMessage)) != true) {
  604. return r;
  605. }
  606. file->ReachableFiles.insert(includedFile->ReachableFiles.begin(),
  607. includedFile->ReachableFiles.end());
  608. return true;
  609. };
  610. std::vector<MacroExpander> macroExpanders;
  611. MacroExpander environmentMacroExpander =
  612. [](const std::string& macroNamespace, const std::string& macroName,
  613. std::string& expanded, int /*version*/) -> ExpandMacroResult {
  614. if (macroNamespace == "penv") {
  615. if (macroName.empty()) {
  616. return ExpandMacroResult::Error;
  617. }
  618. if (cm::optional<std::string> value =
  619. cmSystemTools::GetEnvVar(macroName)) {
  620. expanded += *value;
  621. }
  622. return ExpandMacroResult::Ok;
  623. }
  624. return ExpandMacroResult::Ignore;
  625. };
  626. macroExpanders.push_back(environmentMacroExpander);
  627. for (Json::ArrayIndex i = 0; i < presets.Include.size(); ++i) {
  628. auto include = presets.Include[i];
  629. // Support for macro expansion in includes added in version 7
  630. if (v >= 7) {
  631. if (ExpandMacros(include, macroExpanders, v) != ExpandMacroResult::Ok) {
  632. cmCMakePresetsErrors::INVALID_INCLUDE(&root["include"][i],
  633. &this->parseState);
  634. return false;
  635. }
  636. }
  637. if (!cmSystemTools::FileIsFullPath(include)) {
  638. auto directory = cmSystemTools::GetFilenamePath(filename);
  639. include = cmStrCat(directory, '/', include);
  640. }
  641. if ((result = includeFile(include, rootType, ReadReason::Included,
  642. errMsg)) != true) {
  643. return result;
  644. }
  645. }
  646. if (rootType == RootType::User && readReason == ReadReason::Root) {
  647. auto cmakePresetsFilename = GetFilename(this->SourceDir);
  648. if (cmSystemTools::FileExists(cmakePresetsFilename)) {
  649. if ((result = includeFile(cmakePresetsFilename, RootType::Project,
  650. ReadReason::Root, errMsg)) != true) {
  651. return result;
  652. }
  653. }
  654. }
  655. inProgressFiles.pop_back();
  656. return true;
  657. }