| // Copyright 2017 The Bazel Authors. All rights reserved. | 
 | // | 
 | // Licensed under the Apache License, Version 2.0 (the "License"); | 
 | // you may not use this file except in compliance with the License. | 
 | // You may obtain a copy of the License at | 
 | // | 
 | //    http://www.apache.org/licenses/LICENSE-2.0 | 
 | // | 
 | // Unless required by applicable law or agreed to in writing, software | 
 | // distributed under the License is distributed on an "AS IS" BASIS, | 
 | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
 | // See the License for the specific language governing permissions and | 
 | // limitations under the License. | 
 |  | 
 | #include "src/tools/launcher/python_launcher.h" | 
 |  | 
 | #include <string> | 
 | #include <vector> | 
 |  | 
 | #include "src/main/native/windows/process.h" | 
 | #include "src/tools/launcher/util/launcher_util.h" | 
 |  | 
 | namespace bazel { | 
 | namespace launcher { | 
 |  | 
 | using std::vector; | 
 | using std::wstring; | 
 |  | 
 | static constexpr const char* PYTHON_BIN_PATH = "python_bin_path"; | 
 | static constexpr const char* USE_ZIP_FILE = "use_zip_file"; | 
 | static constexpr const char* PYTHON_FILE_SHORT_PATH = "python_file_short_path"; | 
 |  | 
 | ExitCode PythonBinaryLauncher::Launch() { | 
 |   wstring python_binary = this->GetLaunchInfoByKey(PYTHON_BIN_PATH); | 
 |  | 
 |   // There are three kinds of values for `python_binary`: | 
 |   // 1. An absolute path to a system interpreter. This is the case if | 
 |   // `--python_path` is set by the | 
 |   //    user, or if a `py_runtime` is used that has `interpreter_path` set. | 
 |   // 2. A runfile path to an in-workspace interpreter. This is the case if a | 
 |   // `py_runtime` is used that has `interpreter` set. | 
 |   // 3. The special constant, "python". This is the default case if neither of | 
 |   // the above apply. Rlocation resolves runfiles paths to absolute paths, and | 
 |   // if given an absolute path it leaves it alone, so it's suitable for cases 1 | 
 |   // and 2. | 
 |   if (GetBinaryPathWithoutExtension(python_binary) != L"python") { | 
 |     // Rlocation returns the original path if python_binary is an absolute path. | 
 |     python_binary = this->Rlocation(python_binary, true); | 
 |   } | 
 |  | 
 |   // If specified python binary path doesn't exist, then fall back to | 
 |   // python.exe and hope it's in PATH. | 
 |   if (!DoesFilePathExist(python_binary.c_str())) { | 
 |     python_binary = L"python.exe"; | 
 |   } | 
 |  | 
 |   vector<wstring> args = this->GetCommandlineArguments(); | 
 |   wstring python_file; | 
 |   wstring python_file_short_path = | 
 |       this->GetLaunchInfoByKeyOrEmpty(PYTHON_FILE_SHORT_PATH); | 
 |   if (!python_file_short_path.empty()) { | 
 |     python_file = Rlocation(python_file_short_path, false); | 
 |   } else { | 
 |     wstring use_zip_file = this->GetLaunchInfoByKey(USE_ZIP_FILE); | 
 |     if (use_zip_file == L"1") { | 
 |       python_file = GetBinaryPathWithoutExtension(GetLauncherPath()) + L".zip"; | 
 |     } else { | 
 |       python_file = GetBinaryPathWithoutExtension(GetLauncherPath()); | 
 |     } | 
 |   } | 
 |  | 
 |   // Replace the first argument with python file path | 
 |   // Escaping it, as the python file might contain a " " (eg. When the file is | 
 |   // located under C:\Program Files\...) | 
 |   args[0] = bazel::windows::WindowsEscapeArg(python_file); | 
 |  | 
 |   for (int i = 1; i < args.size(); i++) { | 
 |     args[i] = bazel::windows::WindowsEscapeArg(args[i]); | 
 |   } | 
 |  | 
 |   return this->LaunchProcess(python_binary, args); | 
 | } | 
 |  | 
 | }  // namespace launcher | 
 | }  // namespace bazel |