@typespec/http-client-python 0.4.4 → 0.5.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (156) hide show
  1. package/dist/emitter/emitter.d.ts.map +1 -1
  2. package/dist/emitter/emitter.js +110 -24
  3. package/dist/emitter/emitter.js.map +1 -1
  4. package/dist/emitter/http.js +1 -1
  5. package/dist/emitter/http.js.map +1 -1
  6. package/dist/emitter/lib.d.ts +1 -0
  7. package/dist/emitter/lib.d.ts.map +1 -1
  8. package/dist/emitter/lib.js +1 -0
  9. package/dist/emitter/lib.js.map +1 -1
  10. package/dist/emitter/run-python3.d.ts +2 -0
  11. package/dist/emitter/run-python3.d.ts.map +1 -0
  12. package/dist/emitter/run-python3.js +19 -0
  13. package/dist/emitter/run-python3.js.map +1 -0
  14. package/dist/emitter/system-requirements.d.ts +17 -0
  15. package/dist/emitter/system-requirements.d.ts.map +1 -0
  16. package/dist/emitter/system-requirements.js +167 -0
  17. package/dist/emitter/system-requirements.js.map +1 -0
  18. package/emitter/src/emitter.ts +111 -23
  19. package/emitter/src/http.ts +1 -1
  20. package/emitter/src/lib.ts +2 -0
  21. package/emitter/src/run-python3.ts +20 -0
  22. package/emitter/src/system-requirements.ts +261 -0
  23. package/emitter/temp/tsconfig.tsbuildinfo +1 -1
  24. package/eng/scripts/Test-Packages.ps1 +1 -1
  25. package/eng/scripts/ci/regenerate.ts +20 -8
  26. package/eng/scripts/setup/__pycache__/venvtools.cpython-38.pyc +0 -0
  27. package/eng/scripts/setup/build.ts +16 -0
  28. package/eng/scripts/setup/build_pygen_wheel.py +40 -0
  29. package/eng/scripts/setup/install.py +9 -8
  30. package/eng/scripts/setup/install.ts +12 -0
  31. package/eng/scripts/setup/prepare.py +3 -1
  32. package/eng/scripts/setup/prepare.ts +11 -0
  33. package/eng/scripts/setup/run-python3.ts +1 -6
  34. package/generator/build/lib/pygen/__init__.py +107 -0
  35. package/generator/build/lib/pygen/_version.py +7 -0
  36. package/generator/build/lib/pygen/black.py +71 -0
  37. package/generator/build/lib/pygen/codegen/__init__.py +357 -0
  38. package/generator/build/lib/pygen/codegen/_utils.py +17 -0
  39. package/generator/build/lib/pygen/codegen/models/__init__.py +204 -0
  40. package/generator/build/lib/pygen/codegen/models/base.py +186 -0
  41. package/generator/build/lib/pygen/codegen/models/base_builder.py +118 -0
  42. package/generator/build/lib/pygen/codegen/models/client.py +435 -0
  43. package/generator/build/lib/pygen/codegen/models/code_model.py +237 -0
  44. package/generator/build/lib/pygen/codegen/models/combined_type.py +149 -0
  45. package/generator/build/lib/pygen/codegen/models/constant_type.py +129 -0
  46. package/generator/build/lib/pygen/codegen/models/credential_types.py +214 -0
  47. package/generator/build/lib/pygen/codegen/models/dictionary_type.py +127 -0
  48. package/generator/build/lib/pygen/codegen/models/enum_type.py +238 -0
  49. package/generator/build/lib/pygen/codegen/models/imports.py +291 -0
  50. package/generator/build/lib/pygen/codegen/models/list_type.py +143 -0
  51. package/generator/build/lib/pygen/codegen/models/lro_operation.py +142 -0
  52. package/generator/build/lib/pygen/codegen/models/lro_paging_operation.py +32 -0
  53. package/generator/build/lib/pygen/codegen/models/model_type.py +357 -0
  54. package/generator/build/lib/pygen/codegen/models/operation.py +509 -0
  55. package/generator/build/lib/pygen/codegen/models/operation_group.py +184 -0
  56. package/generator/build/lib/pygen/codegen/models/paging_operation.py +155 -0
  57. package/generator/build/lib/pygen/codegen/models/parameter.py +412 -0
  58. package/generator/build/lib/pygen/codegen/models/parameter_list.py +387 -0
  59. package/generator/build/lib/pygen/codegen/models/primitive_types.py +659 -0
  60. package/generator/build/lib/pygen/codegen/models/property.py +170 -0
  61. package/generator/build/lib/pygen/codegen/models/request_builder.py +189 -0
  62. package/generator/build/lib/pygen/codegen/models/request_builder_parameter.py +115 -0
  63. package/generator/build/lib/pygen/codegen/models/response.py +348 -0
  64. package/generator/build/lib/pygen/codegen/models/utils.py +21 -0
  65. package/generator/build/lib/pygen/codegen/serializers/__init__.py +574 -0
  66. package/generator/build/lib/pygen/codegen/serializers/base_serializer.py +21 -0
  67. package/generator/build/lib/pygen/codegen/serializers/builder_serializer.py +1533 -0
  68. package/generator/build/lib/pygen/codegen/serializers/client_serializer.py +294 -0
  69. package/generator/build/lib/pygen/codegen/serializers/enum_serializer.py +15 -0
  70. package/generator/build/lib/pygen/codegen/serializers/general_serializer.py +213 -0
  71. package/generator/build/lib/pygen/codegen/serializers/import_serializer.py +126 -0
  72. package/generator/build/lib/pygen/codegen/serializers/metadata_serializer.py +198 -0
  73. package/generator/build/lib/pygen/codegen/serializers/model_init_serializer.py +33 -0
  74. package/generator/build/lib/pygen/codegen/serializers/model_serializer.py +335 -0
  75. package/generator/build/lib/pygen/codegen/serializers/operation_groups_serializer.py +89 -0
  76. package/generator/build/lib/pygen/codegen/serializers/operations_init_serializer.py +44 -0
  77. package/generator/build/lib/pygen/codegen/serializers/parameter_serializer.py +221 -0
  78. package/generator/build/lib/pygen/codegen/serializers/patch_serializer.py +19 -0
  79. package/generator/build/lib/pygen/codegen/serializers/request_builders_serializer.py +52 -0
  80. package/generator/build/lib/pygen/codegen/serializers/sample_serializer.py +168 -0
  81. package/generator/build/lib/pygen/codegen/serializers/test_serializer.py +292 -0
  82. package/generator/build/lib/pygen/codegen/serializers/types_serializer.py +31 -0
  83. package/generator/build/lib/pygen/codegen/serializers/utils.py +68 -0
  84. package/generator/build/lib/pygen/codegen/templates/client.py.jinja2 +37 -0
  85. package/generator/build/lib/pygen/codegen/templates/client_container.py.jinja2 +12 -0
  86. package/generator/build/lib/pygen/codegen/templates/config.py.jinja2 +73 -0
  87. package/generator/build/lib/pygen/codegen/templates/config_container.py.jinja2 +16 -0
  88. package/generator/build/lib/pygen/codegen/templates/conftest.py.jinja2 +28 -0
  89. package/generator/build/lib/pygen/codegen/templates/enum.py.jinja2 +13 -0
  90. package/generator/build/lib/pygen/codegen/templates/enum_container.py.jinja2 +10 -0
  91. package/generator/build/lib/pygen/codegen/templates/init.py.jinja2 +24 -0
  92. package/generator/build/lib/pygen/codegen/templates/keywords.jinja2 +27 -0
  93. package/generator/build/lib/pygen/codegen/templates/lro_operation.py.jinja2 +16 -0
  94. package/generator/build/lib/pygen/codegen/templates/lro_paging_operation.py.jinja2 +18 -0
  95. package/generator/build/lib/pygen/codegen/templates/macros.jinja2 +12 -0
  96. package/generator/build/lib/pygen/codegen/templates/metadata.json.jinja2 +167 -0
  97. package/generator/build/lib/pygen/codegen/templates/model_base.py.jinja2 +1174 -0
  98. package/generator/build/lib/pygen/codegen/templates/model_container.py.jinja2 +15 -0
  99. package/generator/build/lib/pygen/codegen/templates/model_dpg.py.jinja2 +97 -0
  100. package/generator/build/lib/pygen/codegen/templates/model_init.py.jinja2 +33 -0
  101. package/generator/build/lib/pygen/codegen/templates/model_msrest.py.jinja2 +92 -0
  102. package/generator/build/lib/pygen/codegen/templates/operation.py.jinja2 +21 -0
  103. package/generator/build/lib/pygen/codegen/templates/operation_group.py.jinja2 +75 -0
  104. package/generator/build/lib/pygen/codegen/templates/operation_groups_container.py.jinja2 +19 -0
  105. package/generator/build/lib/pygen/codegen/templates/operation_tools.jinja2 +81 -0
  106. package/generator/build/lib/pygen/codegen/templates/operations_folder_init.py.jinja2 +17 -0
  107. package/generator/build/lib/pygen/codegen/templates/packaging_templates/CHANGELOG.md.jinja2 +6 -0
  108. package/generator/build/lib/pygen/codegen/templates/packaging_templates/LICENSE.jinja2 +21 -0
  109. package/generator/build/lib/pygen/codegen/templates/packaging_templates/MANIFEST.in.jinja2 +8 -0
  110. package/generator/build/lib/pygen/codegen/templates/packaging_templates/README.md.jinja2 +107 -0
  111. package/generator/build/lib/pygen/codegen/templates/packaging_templates/dev_requirements.txt.jinja2 +9 -0
  112. package/generator/build/lib/pygen/codegen/templates/packaging_templates/setup.py.jinja2 +108 -0
  113. package/generator/build/lib/pygen/codegen/templates/paging_operation.py.jinja2 +21 -0
  114. package/generator/build/lib/pygen/codegen/templates/patch.py.jinja2 +19 -0
  115. package/generator/build/lib/pygen/codegen/templates/pkgutil_init.py.jinja2 +1 -0
  116. package/generator/build/lib/pygen/codegen/templates/request_builder.py.jinja2 +28 -0
  117. package/generator/build/lib/pygen/codegen/templates/request_builders.py.jinja2 +10 -0
  118. package/generator/build/lib/pygen/codegen/templates/rest_init.py.jinja2 +12 -0
  119. package/generator/build/lib/pygen/codegen/templates/sample.py.jinja2 +44 -0
  120. package/generator/build/lib/pygen/codegen/templates/serialization.py.jinja2 +2117 -0
  121. package/generator/build/lib/pygen/codegen/templates/test.py.jinja2 +50 -0
  122. package/generator/build/lib/pygen/codegen/templates/testpreparer.py.jinja2 +26 -0
  123. package/generator/build/lib/pygen/codegen/templates/types.py.jinja2 +7 -0
  124. package/generator/build/lib/pygen/codegen/templates/validation.py.jinja2 +38 -0
  125. package/generator/build/lib/pygen/codegen/templates/vendor.py.jinja2 +96 -0
  126. package/generator/build/lib/pygen/codegen/templates/version.py.jinja2 +4 -0
  127. package/generator/build/lib/pygen/m2r.py +65 -0
  128. package/generator/build/lib/pygen/preprocess/__init__.py +515 -0
  129. package/generator/build/lib/pygen/preprocess/helpers.py +27 -0
  130. package/generator/build/lib/pygen/preprocess/python_mappings.py +226 -0
  131. package/generator/build/lib/pygen/utils.py +163 -0
  132. package/generator/component-detection-pip-report.json +134 -0
  133. package/generator/dev_requirements.txt +0 -1
  134. package/generator/dist/pygen-0.1.0-py3-none-any.whl +0 -0
  135. package/generator/pygen/codegen/__init__.py +4 -4
  136. package/generator/pygen.egg-info/PKG-INFO +7 -4
  137. package/generator/pygen.egg-info/requires.txt +7 -4
  138. package/generator/setup.py +7 -4
  139. package/generator/test/azure/mock_api_tests/asynctests/test_azure_client_generator_core_flatten_async.py +1 -1
  140. package/generator/test/{generic_mock_api_tests/asynctests/test_payload_pageable_async.py → azure/mock_api_tests/asynctests/test_azure_payload_pageable_async.py} +1 -1
  141. package/generator/test/azure/mock_api_tests/conftest.py +5 -4
  142. package/generator/test/azure/mock_api_tests/test_azure_client_generator_core_flatten.py +1 -1
  143. package/generator/test/{generic_mock_api_tests/test_payload_pageable.py → azure/mock_api_tests/test_azure_payload_pageable.py} +1 -1
  144. package/generator/test/{generic_mock_api_tests → azure/mock_api_tests}/test_resiliency_srv_driven.py +4 -2
  145. package/generator/test/{generic_mock_api_tests/asynctests → azure/mock_api_tests}/test_resiliency_srv_driven_async.py +3 -2
  146. package/generator/test/azure/requirements.txt +9 -8
  147. package/generator/test/generic_mock_api_tests/conftest.py +9 -4
  148. package/generator/test/unbranded/mock_api_tests/conftest.py +4 -4
  149. package/generator/test/unbranded/mock_api_tests/test_unbranded.py +1 -1
  150. package/generator/test/unbranded/requirements.txt +1 -8
  151. package/package.json +10 -10
  152. package/generator/requirements.txt +0 -12
  153. /package/generator/test/{generic_mock_api_tests → azure/mock_api_tests}/asynctests/test_client_naming_async.py +0 -0
  154. /package/generator/test/{generic_mock_api_tests → azure/mock_api_tests}/asynctests/test_client_structure_async.py +0 -0
  155. /package/generator/test/{generic_mock_api_tests → azure/mock_api_tests}/test_client_naming.py +0 -0
  156. /package/generator/test/{generic_mock_api_tests → azure/mock_api_tests}/test_client_structure.py +0 -0
@@ -8,10 +8,12 @@ import { EmitContext, NoTarget } from "@typespec/compiler";
8
8
  import { execSync } from "child_process";
9
9
  import fs from "fs";
10
10
  import path, { dirname } from "path";
11
+ import { loadPyodide } from "pyodide";
11
12
  import { fileURLToPath } from "url";
12
13
  import { emitCodeModel } from "./code-model.js";
13
14
  import { saveCodeModelAsYaml } from "./external-process.js";
14
15
  import { PythonEmitterOptions, PythonSdkContext, reportDiagnostic } from "./lib.js";
16
+ import { runPython3 } from "./run-python3.js";
15
17
  import { removeUnderscoresFromNamespace } from "./utils.js";
16
18
 
17
19
  export function getModelsMode(context: SdkContext): "dpg" | "none" {
@@ -85,51 +87,137 @@ export async function $onEmit(context: EmitContext<PythonEmitterOptions>) {
85
87
  });
86
88
  return;
87
89
  }
88
-
89
- addDefaultOptions(sdkContext);
90
90
  const yamlPath = await saveCodeModelAsYaml("python-yaml-path", yamlMap);
91
- let venvPath = path.join(root, "venv");
92
- if (fs.existsSync(path.join(venvPath, "bin"))) {
93
- venvPath = path.join(venvPath, "bin", "python");
94
- } else if (fs.existsSync(path.join(venvPath, "Scripts"))) {
95
- venvPath = path.join(venvPath, "Scripts", "python.exe");
96
- } else {
97
- throw new Error("Virtual environment doesn't exist.");
98
- }
99
- const commandArgs = [
100
- venvPath,
101
- `${root}/eng/scripts/setup/run_tsp.py`,
102
- `--output-folder=${outputDir}`,
103
- `--cadl-file=${yamlPath}`,
104
- ];
91
+ addDefaultOptions(sdkContext);
105
92
  const resolvedOptions = sdkContext.emitContext.options;
93
+ const commandArgs: Record<string, string> = {};
106
94
  if (resolvedOptions["packaging-files-config"]) {
107
95
  const keyValuePairs = Object.entries(resolvedOptions["packaging-files-config"]).map(
108
96
  ([key, value]) => {
109
97
  return `${key}:${value}`;
110
98
  },
111
99
  );
112
- commandArgs.push(`--packaging-files-config='${keyValuePairs.join("|")}'`);
100
+ commandArgs["packaging-files-config"] = keyValuePairs.join("|");
113
101
  resolvedOptions["packaging-files-config"] = undefined;
114
102
  }
115
103
  if (
116
104
  resolvedOptions["package-pprint-name"] !== undefined &&
117
105
  !resolvedOptions["package-pprint-name"].startsWith('"')
118
106
  ) {
119
- resolvedOptions["package-pprint-name"] = `"${resolvedOptions["package-pprint-name"]}"`;
107
+ resolvedOptions["package-pprint-name"] = `${resolvedOptions["package-pprint-name"]}`;
120
108
  }
121
109
 
122
110
  for (const [key, value] of Object.entries(resolvedOptions)) {
123
- commandArgs.push(`--${key}=${value}`);
111
+ commandArgs[key] = value;
124
112
  }
125
113
  if (sdkContext.arm === true) {
126
- commandArgs.push("--azure-arm=true");
114
+ commandArgs["azure-arm"] = "true";
127
115
  }
128
116
  if (resolvedOptions.flavor === "azure") {
129
- commandArgs.push("--emit-cross-language-definition-file=true");
117
+ commandArgs["emit-cross-language-definition-file"] = "true";
130
118
  }
131
- commandArgs.push("--from-typespec=true");
119
+ commandArgs["from-typespec"] = "true";
120
+
132
121
  if (!program.compilerOptions.noEmit && !program.hasError()) {
133
- execSync(commandArgs.join(" "));
122
+ // if not using pyodide and there's no venv, we try to create venv
123
+ if (!resolvedOptions["use-pyodide"] && !fs.existsSync(path.join(root, "venv"))) {
124
+ try {
125
+ await runPython3(path.join(root, "/eng/scripts/setup/install.py"));
126
+ await runPython3(path.join(root, "/eng/scripts/setup/prepare.py"));
127
+ } catch (error) {
128
+ // if the python env is not ready, we use pyodide instead
129
+ resolvedOptions["use-pyodide"] = true;
130
+ }
131
+ }
132
+
133
+ if (resolvedOptions["use-pyodide"]) {
134
+ // here we run with pyodide
135
+ const pyodide = await setupPyodideCall(root);
136
+ // create the output folder if not exists
137
+ if (!fs.existsSync(outputDir)) {
138
+ fs.mkdirSync(outputDir, { recursive: true });
139
+ }
140
+ // mount output folder to pyodide
141
+ pyodide.FS.mkdirTree("/output");
142
+ pyodide.FS.mount(pyodide.FS.filesystems.NODEFS, { root: outputDir }, "/output");
143
+ // mount yaml file to pyodide
144
+ pyodide.FS.mkdirTree("/yaml");
145
+ pyodide.FS.mount(pyodide.FS.filesystems.NODEFS, { root: path.dirname(yamlPath) }, "/yaml");
146
+ const globals = pyodide.toPy({
147
+ outputFolder: "/output",
148
+ yamlFile: `/yaml/${path.basename(yamlPath)}`,
149
+ commandArgs,
150
+ });
151
+ const pythonCode = `
152
+ async def main():
153
+ import warnings
154
+ with warnings.catch_warnings():
155
+ warnings.simplefilter("ignore", SyntaxWarning) # bc of m2r2 dep issues
156
+ from pygen import m2r, preprocess, codegen, black
157
+ m2r.M2R(output_folder=outputFolder, cadl_file=yamlFile, **commandArgs).process()
158
+ preprocess.PreProcessPlugin(output_folder=outputFolder, cadl_file=yamlFile, **commandArgs).process()
159
+ codegen.CodeGenerator(output_folder=outputFolder, cadl_file=yamlFile, **commandArgs).process()
160
+ black.BlackScriptPlugin(output_folder=outputFolder, **commandArgs).process()
161
+
162
+ await main()`;
163
+ await pyodide.runPythonAsync(pythonCode, { globals });
164
+ } else {
165
+ // here we run with native python
166
+ let venvPath = path.join(root, "venv");
167
+ if (fs.existsSync(path.join(venvPath, "bin"))) {
168
+ venvPath = path.join(venvPath, "bin", "python");
169
+ } else if (fs.existsSync(path.join(venvPath, "Scripts"))) {
170
+ venvPath = path.join(venvPath, "Scripts", "python.exe");
171
+ } else {
172
+ throw new Error("Virtual environment doesn't exist.");
173
+ }
174
+ commandArgs["output-folder"] = outputDir;
175
+ commandArgs["cadl-file"] = yamlPath;
176
+ const commandFlags = Object.entries(commandArgs)
177
+ .map(([key, value]) => `--${key}=${value}`)
178
+ .join(" ");
179
+ const command = `${venvPath} ${root}/eng/scripts/setup/run_tsp.py ${commandFlags}`;
180
+ execSync(command);
181
+ }
182
+ }
183
+ }
184
+
185
+ async function setupPyodideCall(root: string) {
186
+ const pyodide = await loadPyodide({
187
+ indexURL: path.dirname(fileURLToPath(import.meta.resolve("pyodide"))),
188
+ });
189
+ const micropipLockPath = path.join(root, "micropip.lock");
190
+ while (true) {
191
+ if (fs.existsSync(micropipLockPath)) {
192
+ try {
193
+ const stats = fs.statSync(micropipLockPath);
194
+ const now = new Date().getTime();
195
+ const lockAge = (now - stats.mtime.getTime()) / 1000;
196
+ if (lockAge > 300) {
197
+ fs.unlinkSync(micropipLockPath);
198
+ }
199
+ } catch (err) {
200
+ // ignore
201
+ }
202
+ }
203
+ try {
204
+ const fd = fs.openSync(micropipLockPath, "wx");
205
+ // mount generator to pyodide
206
+ pyodide.FS.mkdirTree("/generator");
207
+ pyodide.FS.mount(
208
+ pyodide.FS.filesystems.NODEFS,
209
+ { root: path.join(root, "generator") },
210
+ "/generator",
211
+ );
212
+ await pyodide.loadPackage("micropip");
213
+ const micropip = pyodide.pyimport("micropip");
214
+ await micropip.install("emfs:/generator/dist/pygen-0.1.0-py3-none-any.whl");
215
+ fs.closeSync(fd);
216
+ fs.unlinkSync(micropipLockPath);
217
+ break;
218
+ } catch (err) {
219
+ await new Promise((resolve) => setTimeout(resolve, 1000));
220
+ }
134
221
  }
222
+ return pyodide;
135
223
  }
@@ -204,7 +204,7 @@ function emitHttpOperation(
204
204
  apiVersions: [],
205
205
  wantTracing: true,
206
206
  exposeStreamKeyword: true,
207
- crossLanguageDefinitionId: method?.crossLanguageDefintionId,
207
+ crossLanguageDefinitionId: method?.crossLanguageDefinitionId,
208
208
  samples: arrayToRecord(method?.operation.examples),
209
209
  internal: method.access === "internal",
210
210
  };
@@ -17,6 +17,7 @@ export interface PythonEmitterOptions {
17
17
  debug?: boolean;
18
18
  flavor?: "azure";
19
19
  "examples-dir"?: string;
20
+ "use-pyodide"?: boolean;
20
21
  }
21
22
 
22
23
  export interface PythonSdkContext<TServiceOperation extends SdkServiceOperation>
@@ -43,6 +44,7 @@ const EmitterOptionsSchema: JSONSchemaType<PythonEmitterOptions> = {
43
44
  debug: { type: "boolean", nullable: true },
44
45
  flavor: { type: "string", nullable: true },
45
46
  "examples-dir": { type: "string", nullable: true, format: "absolute-path" },
47
+ "use-pyodide": { type: "boolean", nullable: true },
46
48
  },
47
49
  required: [],
48
50
  };
@@ -0,0 +1,20 @@
1
+ // This script wraps logic in @azure-tools/extension to resolve
2
+ // the path to Python 3 so that a Python script file can be run
3
+ // from an npm script in package.json. It uses the same Python 3
4
+ // path resolution algorithm as AutoRest so that the behavior
5
+ // is fully consistent (and also supports AUTOREST_PYTHON_EXE).
6
+ //
7
+ // Invoke it like so: "tsx run-python3.ts script.py"
8
+
9
+ import cp from "child_process";
10
+ import { patchPythonPath } from "./system-requirements.js";
11
+
12
+ export async function runPython3(...args: string[]) {
13
+ const command = await patchPythonPath(["python", ...args], {
14
+ version: ">=3.8",
15
+ environmentVariable: "AUTOREST_PYTHON_EXE",
16
+ });
17
+ cp.execSync(command.join(" "), {
18
+ stdio: [0, 1, 2],
19
+ });
20
+ }
@@ -0,0 +1,261 @@
1
+ import { ChildProcess, spawn, SpawnOptions } from "child_process";
2
+ import { coerce, satisfies } from "semver";
3
+
4
+ /*
5
+ * Copied from @autorest/system-requirements
6
+ */
7
+
8
+ const execute = (
9
+ command: string,
10
+ cmdlineargs: Array<string>,
11
+ options: MoreOptions = {},
12
+ ): Promise<ExecResult> => {
13
+ return new Promise((resolve, reject) => {
14
+ const cp = spawn(command, cmdlineargs, { ...options, stdio: "pipe", shell: true });
15
+ if (options.onCreate) {
16
+ options.onCreate(cp);
17
+ }
18
+
19
+ options.onStdOutData && cp.stdout.on("data", options.onStdOutData);
20
+ options.onStdErrData && cp.stderr.on("data", options.onStdErrData);
21
+
22
+ let err = "";
23
+ let out = "";
24
+ let all = "";
25
+ cp.stderr.on("data", (chunk) => {
26
+ err += chunk;
27
+ all += chunk;
28
+ });
29
+ cp.stdout.on("data", (chunk) => {
30
+ out += chunk;
31
+ all += chunk;
32
+ });
33
+
34
+ cp.on("error", (err) => {
35
+ reject(err);
36
+ });
37
+ cp.on("close", (code, signal) =>
38
+ resolve({
39
+ stdout: out,
40
+ stderr: err,
41
+ log: all,
42
+ error: code ? new Error("Process Failed.") : null,
43
+ code,
44
+ }),
45
+ );
46
+ });
47
+ };
48
+
49
+ const versionIsSatisfied = (version: string, requirement: string): boolean => {
50
+ const cleanedVersion = coerce(version);
51
+ if (!cleanedVersion) {
52
+ throw new Error(`Invalid version ${version}.`);
53
+ }
54
+ return satisfies(cleanedVersion, requirement, true);
55
+ };
56
+
57
+ /**
58
+ * Validate the provided system requirement resolution is satisfying the version requirement if applicable.
59
+ * @param resolution Command resolution.
60
+ * @param actualVersion Version for that resolution.
61
+ * @param requirement Requirement.
62
+ * @returns the resolution if it is valid or an @see SystemRequirementError if not.
63
+ */
64
+ const validateVersionRequirement = (
65
+ resolution: SystemRequirementResolution,
66
+ actualVersion: string,
67
+ requirement: SystemRequirement,
68
+ ): SystemRequirementResolution | SystemRequirementError => {
69
+ if (!requirement.version) {
70
+ return resolution; // No version requirement.
71
+ }
72
+
73
+ try {
74
+ if (versionIsSatisfied(actualVersion, requirement.version)) {
75
+ return resolution;
76
+ }
77
+ return {
78
+ ...resolution,
79
+ error: true,
80
+ message: `'${resolution.command}' version is '${actualVersion}' but doesn't satisfy requirement '${requirement.version}'. Please update.`,
81
+ actualVersion: actualVersion,
82
+ neededVersion: requirement.version,
83
+ };
84
+ } catch {
85
+ return {
86
+ ...resolution,
87
+ error: true,
88
+ message: `Couldn't parse the version ${actualVersion}. This is not a valid semver version.`,
89
+ actualVersion: actualVersion,
90
+ neededVersion: requirement.version,
91
+ };
92
+ }
93
+ };
94
+
95
+ const tryPython = async (
96
+ requirement: SystemRequirement,
97
+ command: string,
98
+ additionalArgs: string[] = [],
99
+ ): Promise<SystemRequirementResolution | SystemRequirementError> => {
100
+ const resolution: SystemRequirementResolution = {
101
+ name: PythonRequirement,
102
+ command,
103
+ additionalArgs: additionalArgs.length > 0 ? additionalArgs : undefined,
104
+ };
105
+
106
+ try {
107
+ const result = await execute(command, [
108
+ ...additionalArgs,
109
+ "-c",
110
+ `"${PRINT_PYTHON_VERSION_SCRIPT}"`,
111
+ ]);
112
+ return validateVersionRequirement(resolution, result.stdout.trim(), requirement);
113
+ } catch (e) {
114
+ return {
115
+ error: true,
116
+ ...resolution,
117
+ message: `'${command}' command line is not found in the path. Make sure to have it installed.`,
118
+ };
119
+ }
120
+ };
121
+
122
+ /**
123
+ * Returns the path to the executable as asked in the requirement.
124
+ * @param requirement System requirement definition.
125
+ * @returns If the requirement provide an environment variable for the path returns the value of that environment variable. undefined otherwise.
126
+ */
127
+ const getExecutablePath = (requirement: SystemRequirement): string | undefined =>
128
+ requirement.environmentVariable && process.env[requirement.environmentVariable];
129
+
130
+ const createPythonErrorMessage = (
131
+ requirement: SystemRequirement,
132
+ errors: SystemRequirementError[],
133
+ ): SystemRequirementError => {
134
+ const versionReq = requirement.version ?? "*";
135
+ const lines = [
136
+ `Couldn't find a valid python interpreter satisfying the requirement (version: ${versionReq}). Tried:`,
137
+ ...errors.map((x) => ` - ${x.command} (${x.message})`),
138
+ ];
139
+
140
+ return {
141
+ error: true,
142
+ name: "python",
143
+ command: "python",
144
+ message: lines.join("\n"),
145
+ };
146
+ };
147
+
148
+ const resolvePythonRequirement = async (
149
+ requirement: SystemRequirement,
150
+ ): Promise<SystemRequirementResolution | SystemRequirementError> => {
151
+ // Hardcoding AUTOREST_PYTHON_EXE is for backward compatibility
152
+ const path = getExecutablePath(requirement) ?? process.env["AUTOREST_PYTHON_EXE"];
153
+ if (path) {
154
+ return await tryPython(requirement, path);
155
+ }
156
+
157
+ const errors: SystemRequirementError[] = [];
158
+ // On windows try `py` executable with `-3` flag.
159
+ if (process.platform === "win32") {
160
+ const pyResult = await tryPython(requirement, "py", ["-3"]);
161
+ if ("error" in pyResult) {
162
+ errors.push(pyResult);
163
+ } else {
164
+ return pyResult;
165
+ }
166
+ }
167
+
168
+ const python3Result = await tryPython(requirement, "python3");
169
+ if ("error" in python3Result) {
170
+ errors.push(python3Result);
171
+ } else {
172
+ return python3Result;
173
+ }
174
+
175
+ const pythonResult = await tryPython(requirement, "python");
176
+ if ("error" in pythonResult) {
177
+ errors.push(pythonResult);
178
+ } else {
179
+ return pythonResult;
180
+ }
181
+
182
+ return createPythonErrorMessage(requirement, errors);
183
+ };
184
+
185
+ /**
186
+ * @param command list of the command and arguments. First item in array must be a python exe @see KnownPythonExe. (e.g. ["python", "my_python_file.py"]
187
+ * @param requirement
188
+ */
189
+ export const patchPythonPath = async (
190
+ command: PythonCommandLine,
191
+ requirement: SystemRequirement,
192
+ ): Promise<string[]> => {
193
+ const [_, ...args] = command;
194
+ const resolution = await resolvePythonRequirement(requirement);
195
+ if ("error" in resolution) {
196
+ throw new Error(`Failed to find compatible python version. ${resolution.message}`);
197
+ }
198
+ return [resolution.command, ...(resolution.additionalArgs ?? []), ...args];
199
+ };
200
+
201
+ ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
202
+ // TYPES
203
+ const PythonRequirement = "python";
204
+ const PRINT_PYTHON_VERSION_SCRIPT = "import sys; print('.'.join(map(str, sys.version_info[:3])))";
205
+
206
+ type KnownPythonExe = "python.exe" | "python3.exe" | "python" | "python3";
207
+ type PythonCommandLine = [KnownPythonExe, ...string[]];
208
+
209
+ interface MoreOptions extends SpawnOptions {
210
+ onCreate?(cp: ChildProcess): void;
211
+ onStdOutData?(chunk: any): void;
212
+ onStdErrData?(chunk: any): void;
213
+ }
214
+
215
+ interface SystemRequirement {
216
+ version?: string;
217
+ /**
218
+ * Name of an environment variable where the user could provide the path to the exe.
219
+ * @example "AUTOREST_PYTHON_PATH"
220
+ */
221
+ environmentVariable?: string;
222
+ }
223
+
224
+ interface SystemRequirementResolution {
225
+ /**
226
+ * Name of the requirement.
227
+ * @example python, java, etc.
228
+ */
229
+ name: string;
230
+
231
+ /**
232
+ * Name of the command
233
+ * @example python3, /home/my_user/python39/python, java, etc.
234
+ */
235
+ command: string;
236
+
237
+ /**
238
+ * List of additional arguments to pass to this command.
239
+ * @example '-3' for 'py' to specify to use python 3
240
+ */
241
+ additionalArgs?: string[];
242
+ }
243
+
244
+ interface ExecResult {
245
+ stdout: string;
246
+ stderr: string;
247
+
248
+ /**
249
+ * Union of stdout and stderr.
250
+ */
251
+ log: string;
252
+ error: Error | null;
253
+ code: number | null;
254
+ }
255
+
256
+ interface SystemRequirementError extends SystemRequirementResolution {
257
+ error: true;
258
+ message: string;
259
+ neededVersion?: string;
260
+ actualVersion?: string;
261
+ }