cmCMakePresetsGraphReadJSON.cxx 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750
  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 "cmCMakePresetErrors.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 = 7;
  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. cmCMakePresetErrors::INVALID_CONDITION, ConditionStringHelper);
  64. auto const ConstConditionHelper =
  65. JSONHelperBuilder::Object<cmCMakePresetsGraphInternal::ConstCondition>(
  66. cmCMakePresetErrors::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. cmCMakePresetErrors::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. cmCMakePresetErrors::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. cmCMakePresetErrors::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. cmCMakePresetErrors::INVALID_CONDITION, SubConditionHelper);
  99. auto const AnyAllOfConditionHelper =
  100. JSONHelperBuilder::Object<cmCMakePresetsGraphInternal::AnyAllOfCondition>(
  101. cmCMakePresetErrors::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. cmCMakePresetErrors::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. cmCMakePresetErrors::INVALID_CONDITION(value, state);
  133. return false;
  134. }
  135. if (!(*value)["type"].isString()) {
  136. cmCMakePresetErrors::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. cmCMakePresetErrors::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. cmCMakePresetErrors::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. cmCMakePresetErrors::INVALID_PRESET(value, state);
  216. return false;
  217. }
  218. auto const VersionIntHelper =
  219. JSONHelperBuilder::Int(cmCMakePresetErrors::INVALID_VERSION);
  220. auto const VersionHelper = JSONHelperBuilder::Required<int>(
  221. cmCMakePresetErrors::NO_VERSION, VersionIntHelper);
  222. auto const RootVersionHelper =
  223. JSONHelperBuilder::Object<int>(cmCMakePresetErrors::INVALID_ROOT_OBJECT)
  224. .Bind("version"_s, VersionHelper, false);
  225. auto const CMakeVersionUIntHelper =
  226. JSONHelperBuilder::UInt(cmCMakePresetErrors::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(cmCMakePresetErrors::INVALID_INCLUDE);
  235. auto const IncludeVectorHelper = JSONHelperBuilder::Vector<std::string>(
  236. cmCMakePresetErrors::INVALID_INCLUDE, IncludeHelper);
  237. auto const RootPresetsHelper =
  238. JSONHelperBuilder::Object<RootPresets>(
  239. cmCMakePresetErrors::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. cmCMakePresetErrors::INVALID_ROOT),
  257. false);
  258. }
  259. namespace cmCMakePresetsGraphInternal {
  260. bool PresetStringHelper(std::string& out, const Json::Value* value,
  261. cmJSONState* state)
  262. {
  263. static auto const helper = JSONHelperBuilder::String();
  264. return helper(out, value, state);
  265. }
  266. bool PresetNameHelper(std::string& out, const Json::Value* value,
  267. cmJSONState* state)
  268. {
  269. if (!value || !value->isString() || value->asString().empty()) {
  270. cmCMakePresetErrors::INVALID_PRESET_NAME(value, state);
  271. return false;
  272. }
  273. out = value->asString();
  274. return true;
  275. }
  276. bool PresetVectorStringHelper(std::vector<std::string>& out,
  277. const Json::Value* value, cmJSONState* state)
  278. {
  279. static auto const helper = JSONHelperBuilder::Vector<std::string>(
  280. cmCMakePresetErrors::INVALID_PRESET,
  281. cmCMakePresetsGraphInternal::PresetStringHelper);
  282. return helper(out, value, state);
  283. }
  284. bool PresetBoolHelper(bool& out, const Json::Value* value, cmJSONState* state)
  285. {
  286. static auto const helper = JSONHelperBuilder::Bool();
  287. return helper(out, value, state);
  288. }
  289. bool PresetOptionalBoolHelper(cm::optional<bool>& out,
  290. const Json::Value* value, cmJSONState* state)
  291. {
  292. static auto const helper =
  293. JSONHelperBuilder::Optional<bool>(PresetBoolHelper);
  294. return helper(out, value, state);
  295. }
  296. bool PresetIntHelper(int& out, const Json::Value* value, cmJSONState* state)
  297. {
  298. static auto const helper = JSONHelperBuilder::Int();
  299. return helper(out, value, state);
  300. }
  301. bool PresetOptionalIntHelper(cm::optional<int>& out, const Json::Value* value,
  302. cmJSONState* state)
  303. {
  304. static auto const helper = JSONHelperBuilder::Optional<int>(PresetIntHelper);
  305. return helper(out, value, state);
  306. }
  307. bool PresetVectorIntHelper(std::vector<int>& out, const Json::Value* value,
  308. cmJSONState* state)
  309. {
  310. static auto const helper = JSONHelperBuilder::Vector<int>(
  311. cmCMakePresetErrors::INVALID_PRESET, PresetIntHelper);
  312. return helper(out, value, state);
  313. }
  314. cmJSONHelper<std::nullptr_t> VendorHelper(const ErrorGenerator& error)
  315. {
  316. return [error](std::nullptr_t& /*out*/, const Json::Value* value,
  317. cmJSONState* state) -> bool {
  318. if (!value) {
  319. return true;
  320. }
  321. if (!value->isObject()) {
  322. error(value, state);
  323. return false;
  324. }
  325. return true;
  326. };
  327. }
  328. bool PresetConditionHelper(
  329. std::shared_ptr<cmCMakePresetsGraph::Condition>& out,
  330. const Json::Value* value, cmJSONState* state)
  331. {
  332. std::unique_ptr<cmCMakePresetsGraph::Condition> ptr;
  333. auto result = ConditionHelper(ptr, value, state);
  334. out = std::move(ptr);
  335. return result;
  336. }
  337. bool PresetVectorOneOrMoreStringHelper(std::vector<std::string>& out,
  338. const Json::Value* value,
  339. cmJSONState* state)
  340. {
  341. out.clear();
  342. if (!value) {
  343. return true;
  344. }
  345. if (value->isString()) {
  346. out.push_back(value->asString());
  347. return true;
  348. }
  349. return PresetVectorStringHelper(out, value, state);
  350. }
  351. bool EnvironmentMapHelper(
  352. std::map<std::string, cm::optional<std::string>>& out,
  353. const Json::Value* value, cmJSONState* state)
  354. {
  355. static auto const helper = JSONHelperBuilder::Map<cm::optional<std::string>>(
  356. cmCMakePresetErrors::INVALID_PRESET, EnvironmentHelper);
  357. return helper(out, value, state);
  358. }
  359. }
  360. bool cmCMakePresetsGraph::ReadJSONFile(const std::string& filename,
  361. RootType rootType,
  362. ReadReason readReason,
  363. std::vector<File*>& inProgressFiles,
  364. File*& file, std::string& errMsg)
  365. {
  366. bool result;
  367. for (auto const& f : this->Files) {
  368. if (cmSystemTools::SameFile(filename, f->Filename)) {
  369. file = f.get();
  370. auto fileIt =
  371. std::find(inProgressFiles.begin(), inProgressFiles.end(), file);
  372. if (fileIt != inProgressFiles.end()) {
  373. cmCMakePresetErrors::CYCLIC_INCLUDE(filename, &this->parseState);
  374. return false;
  375. }
  376. return true;
  377. }
  378. }
  379. Json::Value root;
  380. this->parseState = cmJSONState(filename, &root);
  381. if (!this->parseState.errors.empty()) {
  382. return false;
  383. }
  384. int v = 0;
  385. if ((result = RootVersionHelper(v, &root, &parseState)) != true) {
  386. return result;
  387. }
  388. if (v < MIN_VERSION || v > MAX_VERSION) {
  389. cmCMakePresetErrors::UNRECOGNIZED_VERSION(&root["version"],
  390. &this->parseState);
  391. return false;
  392. }
  393. // Support for build and test presets added in version 2.
  394. if (v < 2) {
  395. if (root.isMember("buildPresets")) {
  396. cmCMakePresetErrors::BUILD_TEST_PRESETS_UNSUPPORTED(
  397. &root["buildPresets"], &this->parseState);
  398. return false;
  399. }
  400. if (root.isMember("testPresets")) {
  401. cmCMakePresetErrors::BUILD_TEST_PRESETS_UNSUPPORTED(&root["testPresets"],
  402. &this->parseState);
  403. return false;
  404. }
  405. }
  406. // Support for package presets added in version 6.
  407. if (v < 6 && root.isMember("packagePresets")) {
  408. cmCMakePresetErrors::PACKAGE_PRESETS_UNSUPPORTED(&root["packagePresets"],
  409. &this->parseState);
  410. return false;
  411. }
  412. // Support for workflow presets added in version 6.
  413. if (v < 6 && root.isMember("workflowPresets")) {
  414. cmCMakePresetErrors::WORKFLOW_PRESETS_UNSUPPORTED(&root["workflowPresets"],
  415. &this->parseState);
  416. return false;
  417. }
  418. // Support for include added in version 4.
  419. if (v < 4 && root.isMember("include")) {
  420. cmCMakePresetErrors::INCLUDE_UNSUPPORTED(&root["include"],
  421. &this->parseState);
  422. return false;
  423. }
  424. RootPresets presets;
  425. if ((result = RootPresetsHelper(presets, &root, &parseState)) != true) {
  426. return result;
  427. }
  428. unsigned int currentMajor = cmVersion::GetMajorVersion();
  429. unsigned int currentMinor = cmVersion::GetMinorVersion();
  430. unsigned int currentPatch = cmVersion::GetPatchVersion();
  431. auto const& required = presets.CMakeMinimumRequired;
  432. if (required.Major > currentMajor) {
  433. ErrorGenerator error = cmCMakePresetErrors::UNRECOGNIZED_CMAKE_VERSION(
  434. "major", currentMajor, required.Major);
  435. error(&root["cmakeMinimumRequired"]["major"], &this->parseState);
  436. return false;
  437. }
  438. if (required.Major == currentMajor) {
  439. if (required.Minor > currentMinor) {
  440. ErrorGenerator error = cmCMakePresetErrors::UNRECOGNIZED_CMAKE_VERSION(
  441. "minor", currentMinor, required.Minor);
  442. error(&root["cmakeMinimumRequired"]["minor"], &this->parseState);
  443. return false;
  444. }
  445. if (required.Minor == currentMinor && required.Patch > currentPatch) {
  446. ErrorGenerator error = cmCMakePresetErrors::UNRECOGNIZED_CMAKE_VERSION(
  447. "patch", currentPatch, required.Patch);
  448. error(&root["cmakeMinimumRequired"]["patch"], &this->parseState);
  449. return false;
  450. }
  451. }
  452. auto filePtr = cm::make_unique<File>();
  453. file = filePtr.get();
  454. this->Files.emplace_back(std::move(filePtr));
  455. inProgressFiles.emplace_back(file);
  456. file->Filename = filename;
  457. file->Version = v;
  458. file->ReachableFiles.insert(file);
  459. for (auto& preset : presets.ConfigurePresets) {
  460. preset.OriginFile = file;
  461. if (preset.Name.empty()) {
  462. // No error, already handled by PresetNameHelper
  463. return false;
  464. }
  465. PresetPair<ConfigurePreset> presetPair;
  466. presetPair.Unexpanded = preset;
  467. presetPair.Expanded = cm::nullopt;
  468. if (!this->ConfigurePresets.emplace(preset.Name, presetPair).second) {
  469. cmCMakePresetErrors::DUPLICATE_PRESETS(preset.Name, &this->parseState);
  470. return false;
  471. }
  472. // Support for installDir presets added in version 3.
  473. if (v < 3 && !preset.InstallDir.empty()) {
  474. cmCMakePresetErrors::INSTALL_PREFIX_UNSUPPORTED(&root["installDir"],
  475. &this->parseState);
  476. return false;
  477. }
  478. // Support for conditions added in version 3.
  479. if (v < 3 && preset.ConditionEvaluator) {
  480. cmCMakePresetErrors::CONDITION_UNSUPPORTED(&this->parseState);
  481. return false;
  482. }
  483. // Support for toolchainFile presets added in version 3.
  484. if (v < 3 && !preset.ToolchainFile.empty()) {
  485. cmCMakePresetErrors::TOOLCHAIN_FILE_UNSUPPORTED(&this->parseState);
  486. return false;
  487. }
  488. // Support for trace presets added in version 7.
  489. if (v < 7 &&
  490. (preset.TraceMode.has_value() || preset.TraceFormat.has_value() ||
  491. !preset.TraceRedirect.empty() || !preset.TraceSource.empty())) {
  492. cmCMakePresetErrors::TRACE_UNSUPPORTED(&this->parseState);
  493. return false;
  494. }
  495. this->ConfigurePresetOrder.push_back(preset.Name);
  496. }
  497. for (auto& preset : presets.BuildPresets) {
  498. preset.OriginFile = file;
  499. if (preset.Name.empty()) {
  500. // No error, already handled by PresetNameHelper
  501. return false;
  502. }
  503. PresetPair<BuildPreset> presetPair;
  504. presetPair.Unexpanded = preset;
  505. presetPair.Expanded = cm::nullopt;
  506. if (!this->BuildPresets.emplace(preset.Name, presetPair).second) {
  507. cmCMakePresetErrors::DUPLICATE_PRESETS(preset.Name, &this->parseState);
  508. return false;
  509. }
  510. // Support for conditions added in version 3.
  511. if (v < 3 && preset.ConditionEvaluator) {
  512. cmCMakePresetErrors::CONDITION_UNSUPPORTED(&this->parseState);
  513. return false;
  514. }
  515. this->BuildPresetOrder.push_back(preset.Name);
  516. }
  517. for (auto& preset : presets.TestPresets) {
  518. preset.OriginFile = file;
  519. if (preset.Name.empty()) {
  520. // No error, already handled by PresetNameHelper
  521. return false;
  522. }
  523. PresetPair<TestPreset> presetPair;
  524. presetPair.Unexpanded = preset;
  525. presetPair.Expanded = cm::nullopt;
  526. if (!this->TestPresets.emplace(preset.Name, presetPair).second) {
  527. cmCMakePresetErrors::DUPLICATE_PRESETS(preset.Name, &this->parseState);
  528. return false;
  529. }
  530. // Support for conditions added in version 3.
  531. if (v < 3 && preset.ConditionEvaluator) {
  532. cmCMakePresetErrors::CONDITION_UNSUPPORTED(&this->parseState);
  533. return false;
  534. }
  535. // Support for TestOutputTruncation added in version 5.
  536. if (v < 5 && preset.Output && preset.Output->TestOutputTruncation) {
  537. cmCMakePresetErrors::TEST_OUTPUT_TRUNCATION_UNSUPPORTED(
  538. &this->parseState);
  539. return false;
  540. }
  541. // Support for outputJUnitFile added in version 6.
  542. if (v < 6 && preset.Output && !preset.Output->OutputJUnitFile.empty()) {
  543. cmCMakePresetErrors::CTEST_JUNIT_UNSUPPORTED(&this->parseState);
  544. return false;
  545. }
  546. this->TestPresetOrder.push_back(preset.Name);
  547. }
  548. for (auto& preset : presets.PackagePresets) {
  549. preset.OriginFile = file;
  550. if (preset.Name.empty()) {
  551. // No error, already handled by PresetNameHelper
  552. return false;
  553. }
  554. PresetPair<PackagePreset> presetPair;
  555. presetPair.Unexpanded = preset;
  556. presetPair.Expanded = cm::nullopt;
  557. if (!this->PackagePresets.emplace(preset.Name, presetPair).second) {
  558. cmCMakePresetErrors::DUPLICATE_PRESETS(preset.Name, &this->parseState);
  559. return false;
  560. }
  561. // Support for conditions added in version 3, but this requires version 5
  562. // already, so no action needed.
  563. this->PackagePresetOrder.push_back(preset.Name);
  564. }
  565. for (auto& preset : presets.WorkflowPresets) {
  566. preset.OriginFile = file;
  567. if (preset.Name.empty()) {
  568. // No error, already handled by PresetNameHelper
  569. return false;
  570. }
  571. PresetPair<WorkflowPreset> presetPair;
  572. presetPair.Unexpanded = preset;
  573. presetPair.Expanded = cm::nullopt;
  574. if (!this->WorkflowPresets.emplace(preset.Name, presetPair).second) {
  575. cmCMakePresetErrors::DUPLICATE_PRESETS(preset.Name, &this->parseState);
  576. return false;
  577. }
  578. // Support for conditions added in version 3, but this requires version 6
  579. // already, so no action needed.
  580. this->WorkflowPresetOrder.push_back(preset.Name);
  581. }
  582. auto const includeFile = [this, &inProgressFiles,
  583. file](const std::string& include,
  584. RootType rootType2, ReadReason readReason2,
  585. std::string& FailureMessage) -> bool {
  586. bool r;
  587. File* includedFile;
  588. if ((r =
  589. this->ReadJSONFile(include, rootType2, readReason2, inProgressFiles,
  590. includedFile, FailureMessage)) != true) {
  591. return r;
  592. }
  593. file->ReachableFiles.insert(includedFile->ReachableFiles.begin(),
  594. includedFile->ReachableFiles.end());
  595. return true;
  596. };
  597. std::vector<MacroExpander> macroExpanders;
  598. MacroExpander environmentMacroExpander =
  599. [](const std::string& macroNamespace, const std::string& macroName,
  600. std::string& expanded, int /*version*/) -> ExpandMacroResult {
  601. if (macroNamespace == "penv") {
  602. if (macroName.empty()) {
  603. return ExpandMacroResult::Error;
  604. }
  605. if (cm::optional<std::string> value =
  606. cmSystemTools::GetEnvVar(macroName)) {
  607. expanded += *value;
  608. }
  609. return ExpandMacroResult::Ok;
  610. }
  611. return ExpandMacroResult::Ignore;
  612. };
  613. macroExpanders.push_back(environmentMacroExpander);
  614. for (Json::ArrayIndex i = 0; i < presets.Include.size(); ++i) {
  615. auto include = presets.Include[i];
  616. // Support for macro expansion in includes added in version 7
  617. if (v >= 7) {
  618. if (ExpandMacros(include, macroExpanders, v) != ExpandMacroResult::Ok) {
  619. cmCMakePresetErrors::INVALID_INCLUDE(&root["include"][i],
  620. &this->parseState);
  621. return false;
  622. }
  623. }
  624. if (!cmSystemTools::FileIsFullPath(include)) {
  625. auto directory = cmSystemTools::GetFilenamePath(filename);
  626. include = cmStrCat(directory, '/', include);
  627. }
  628. if ((result = includeFile(include, rootType, ReadReason::Included,
  629. errMsg)) != true) {
  630. return result;
  631. }
  632. }
  633. if (rootType == RootType::User && readReason == ReadReason::Root) {
  634. auto cmakePresetsFilename = GetFilename(this->SourceDir);
  635. if (cmSystemTools::FileExists(cmakePresetsFilename)) {
  636. if ((result = includeFile(cmakePresetsFilename, RootType::Project,
  637. ReadReason::Root, errMsg)) != true) {
  638. return result;
  639. }
  640. }
  641. }
  642. inProgressFiles.pop_back();
  643. return true;
  644. }