浏览代码

cmCTestRunTest: Implement the ENVIRONMENT test property with EnvDiff too

Going through the same internal API for both `ENVIRONMENT` and
`ENVIRONMENT_MODIFICATION` properties will make it easier to implement
checkpointing for `MYVAR=reset:` more efficiently if the need ever
presents itself.  It also makes the two-stage nature of the environment
mutation clearer in the code itself.
Alex Reinking 3 年之前
父节点
当前提交
e2854b4fa2
共有 3 个文件被更改,包括 34 次插入13 次删除
  1. 7 8
      Source/CTest/cmCTestRunTest.cxx
  2. 18 5
      Source/cmSystemTools.cxx
  3. 9 0
      Source/cmSystemTools.h

+ 7 - 8
Source/CTest/cmCTestRunTest.cxx

@@ -784,16 +784,15 @@ bool cmCTestRunTest::ForkProcess(
   this->TestProcess->SetTimeout(timeout);
 
   cmSystemTools::SaveRestoreEnvironment sre;
-
   std::ostringstream envMeasurement;
+
+  // We split processing ENVIRONMENT and ENVIRONMENT_MODIFICATION into two
+  // phases to ensure that MYVAR=reset: in the latter phase resets to the
+  // former phase's settings, rather than to the original environment.
   if (environment && !environment->empty()) {
-    // Environment modification works on the assumption that the environment is
-    // actually modified here. If another strategy is used, there will need to
-    // be updates in `EnvDiff::ParseOperation`.
-    cmSystemTools::AppendEnv(*environment);
-    for (auto const& var : *environment) {
-      envMeasurement << var << std::endl;
-    }
+    cmSystemTools::EnvDiff diff;
+    diff.AppendEnv(*environment);
+    diff.ApplyToCurrentEnv(&envMeasurement);
   }
 
   if (environment_modification && !environment_modification->empty()) {

+ 18 - 5
Source/cmSystemTools.cxx

@@ -1564,6 +1564,24 @@ void cmSystemTools::AppendEnv(std::vector<std::string> const& env)
   }
 }
 
+void cmSystemTools::EnvDiff::AppendEnv(std::vector<std::string> const& env)
+{
+  for (std::string const& eit : env) {
+    this->PutEnv(eit);
+  }
+}
+
+void cmSystemTools::EnvDiff::PutEnv(const std::string& env)
+{
+  auto const eq_loc = env.find('=');
+  if (eq_loc != std::string::npos) {
+    std::string name = env.substr(0, eq_loc);
+    diff[name] = env.substr(eq_loc + 1);
+  } else {
+    diff[env] = {};
+  }
+}
+
 bool cmSystemTools::EnvDiff::ParseOperation(const std::string& envmod)
 {
   char path_sep = GetSystemPathlistSeparator();
@@ -1575,11 +1593,6 @@ bool cmSystemTools::EnvDiff::ParseOperation(const std::string& envmod)
     if (old_value) {
       output = *old_value;
     } else {
-      // This only works because the environment is actually modified when
-      // processing the ENVIRONMENT property in CTest and cmake -E env
-      // (`AppendEnv`). If either one ever just creates an environment block
-      // directly, that block will need to be queried for the subprocess'
-      // value instead.
       const char* curval = cmSystemTools::GetEnv(name);
       if (curval) {
         output = curval;

+ 9 - 0
Source/cmSystemTools.h

@@ -388,6 +388,15 @@ public:
   class EnvDiff
   {
   public:
+    /** Append multiple variables to the current environment diff */
+    void AppendEnv(std::vector<std::string> const& env);
+
+    /**
+     * Add a single variable (or remove if no = sign) to the current
+     * environment diff.
+     */
+    void PutEnv(const std::string& env);
+
     /**
      * Apply an ENVIRONMENT_MODIFICATION operation to this diff. Returns
      * false and issues an error on parse failure.