From d1e442670673e18bea0a341af73a51f3678f020f Mon Sep 17 00:00:00 2001 From: Matt McCormick Date: Tue, 26 Nov 2024 13:05:19 -0500 Subject: [PATCH 1/8] feat(bindgen): support string choices Future todo: handle these as string enums. --- .../itk-wasm/src/bindgen/canonical-type.js | 5 +- .../bindgen/python/function-module-args.js | 19 +++--- .../python/wasi/wasi-function-module.js | 62 ++++++++++++------- .../src/bindgen/typescript/function-module.js | 14 ++++- .../input-output-json-test.cxx | 3 + 5 files changed, 69 insertions(+), 34 deletions(-) diff --git a/packages/core/typescript/itk-wasm/src/bindgen/canonical-type.js b/packages/core/typescript/itk-wasm/src/bindgen/canonical-type.js index c3701398c..b6ecab2e0 100644 --- a/packages/core/typescript/itk-wasm/src/bindgen/canonical-type.js +++ b/packages/core/typescript/itk-wasm/src/bindgen/canonical-type.js @@ -1,7 +1,8 @@ function canonicalType(parameterType) { // Strip extras - const canonical = parameterType.split(' ')[0] + let canonical = parameterType.split(' ')[0] + canonical = canonical.split(':')[0] return canonical } -export default canonicalType \ No newline at end of file +export default canonicalType diff --git a/packages/core/typescript/itk-wasm/src/bindgen/python/function-module-args.js b/packages/core/typescript/itk-wasm/src/bindgen/python/function-module-args.js index 9414f8c12..9ee168b7f 100644 --- a/packages/core/typescript/itk-wasm/src/bindgen/python/function-module-args.js +++ b/packages/core/typescript/itk-wasm/src/bindgen/python/function-module-args.js @@ -5,13 +5,15 @@ import interfaceJsonTypeToInterfaceType from '../interface-json-type-to-interfac import canonicalType from '../canonical-type.js' function functionModuleArgs(interfaceJson) { - let functionArgs = "" + let functionArgs = '' interfaceJson['inputs'].forEach((value) => { const canonical = canonicalType(value.type) const pythonType = interfaceJsonTypeToPythonType.get(canonical) functionArgs += ` ${snakeCase(value.name)}: ${pythonType},\n` }) - const outputFiles = interfaceJson.outputs.filter(o => { return o.type.includes('FILE') }) + const outputFiles = interfaceJson.outputs.filter((o) => { + return o.type.includes('FILE') + }) outputFiles.forEach((output) => { const isArray = output.itemsExpectedMax > 1 const optionName = `${output.name}` @@ -22,19 +24,19 @@ function functionModuleArgs(interfaceJson) { } }) interfaceJson['parameters'].forEach((value) => { - if (value.name === "memory-io" || value.name === "version") { + if (value.name === 'memory-io' || value.name === 'version') { return } const canonical = canonicalType(value.type) const pythonType = interfaceJsonTypeToPythonType.get(canonical) if (interfaceJsonTypeToInterfaceType.has(value.type)) { - if(value.required && value.itemsExpectedMax > 1) { + if (value.required && value.itemsExpectedMax > 1) { functionArgs += ` ${snakeCase(value.name)}: List[${pythonType}] = [],\n` } else { functionArgs += ` ${snakeCase(value.name)}: Optional[${pythonType}] = None,\n` } } else { - if(value.itemsExpectedMax > 1) { + if (value.itemsExpectedMax > 1) { if (value.required) { functionArgs += ` ${snakeCase(value.name)}: List[${pythonType}]` } else { @@ -42,17 +44,16 @@ function functionModuleArgs(interfaceJson) { } } else { functionArgs += ` ${snakeCase(value.name)}: ${pythonType}` - } - if(value.type === "BOOL") { + if (value.type === 'BOOL') { functionArgs += ` = False,\n` - } else if(value.type === "TEXT") { + } else if (value.type.startsWith('TEXT')) { if (value.default) { functionArgs += ` = "${value.default}",\n` } else { functionArgs += ` = "",\n` } - } else if(value.required && value.itemsExpectedMax > 1) { + } else if (value.required && value.itemsExpectedMax > 1) { functionArgs += ` = [],\n` } else { if (value.itemsExpectedMax > 1) { diff --git a/packages/core/typescript/itk-wasm/src/bindgen/python/wasi/wasi-function-module.js b/packages/core/typescript/itk-wasm/src/bindgen/python/wasi/wasi-function-module.js index 322225957..ec3f3f407 100644 --- a/packages/core/typescript/itk-wasm/src/bindgen/python/wasi/wasi-function-module.js +++ b/packages/core/typescript/itk-wasm/src/bindgen/python/wasi/wasi-function-module.js @@ -53,8 +53,8 @@ from itkwasm import ( const interfaceType = interfaceJsonTypeToInterfaceType.get(output.type) const isArray = output.itemsExpectedMax > 1 switch (interfaceType) { - case "TextFile": - case "BinaryFile": + case 'TextFile': + case 'BinaryFile': if (isArray) { haveArray = true pipelineOutputs += ` *${snakeCase(output.name)}_pipeline_outputs,\n` @@ -97,12 +97,12 @@ from itkwasm import ( if (interfaceJsonTypeToInterfaceType.has(input.type)) { const interfaceType = interfaceJsonTypeToInterfaceType.get(input.type) switch (interfaceType) { - case "TextFile": - case "BinaryFile": + case 'TextFile': + case 'BinaryFile': pipelineInputs += ` PipelineInput(InterfaceTypes.${interfaceType}, ${interfaceType}(PurePosixPath(${snakeCase(input.name)}))),\n` break - case "TextStream": - case "BinaryStream": + case 'TextStream': + case 'BinaryStream': pipelineInputs += ` PipelineInput(InterfaceTypes.${interfaceType}, ${interfaceType}(${snakeCase(input.name)})),\n` break default: @@ -113,7 +113,7 @@ from itkwasm import ( let args = ` args: List[str] = ['--memory-io',]\n` let inputCount = 0 - args += " # Inputs\n" + args += ' # Inputs\n' interfaceJson.inputs.forEach((input) => { const snakeName = snakeCase(input.name) if (interfaceJsonTypeToInterfaceType.has(input.type)) { @@ -122,7 +122,9 @@ from itkwasm import ( args += ` if not Path(${snakeName}).exists():\n` args += ` raise FileNotFoundError("${snakeName} does not exist")\n` } - const name = interfaceType.includes('File') ? `str(PurePosixPath(${snakeName}))` : `'${inputCount.toString()}'` + const name = interfaceType.includes('File') + ? `str(PurePosixPath(${snakeName}))` + : `'${inputCount.toString()}'` args += ` args.append(${name})\n` inputCount++ } else { @@ -131,7 +133,7 @@ from itkwasm import ( }) let outputCount = 0 - args += " # Outputs\n" + args += ' # Outputs\n' interfaceJson.outputs.forEach((output) => { const snake = snakeCase(output.name) if (interfaceJsonTypeToInterfaceType.has(output.type)) { @@ -158,7 +160,7 @@ from itkwasm import ( } }) - args += " # Options\n" + args += ' # Options\n' args += ` input_count = len(pipeline_inputs)\n` interfaceJson.parameters.forEach((parameter) => { if (parameter.name === 'memory-io' || parameter.name === 'version') { @@ -166,7 +168,7 @@ from itkwasm import ( return } const snake = snakeCase(parameter.name) - if (parameter.type === "BOOL") { + if (parameter.type === 'BOOL') { args += ` if ${snake}:\n` args += ` args.append('--${parameter.name}')\n` } else if (parameter.itemsExpectedMax > 1) { @@ -177,7 +179,9 @@ from itkwasm import ( args += ` args.append('--${parameter.name}')\n` args += ` for value in ${snake}:\n` if (interfaceJsonTypeToInterfaceType.has(parameter.type)) { - const interfaceType = interfaceJsonTypeToInterfaceType.get(parameter.type) + const interfaceType = interfaceJsonTypeToInterfaceType.get( + parameter.type + ) if (interfaceType.includes('File')) { // for files args += ` input_file = str(PurePosixPath(value))\n` @@ -195,12 +199,19 @@ from itkwasm import ( args += ` input_count += 1\n` } } else { + if (parameter.type.startsWith('TEXT:{')) { + const choices = parameter.type.split('{')[1].split('}')[0].split(',') + args += ` if ${snake} not in (${choices.map((c) => `'${c}'`).join(',')}):\n` + args += ` raise ValueError(f'${snake} must be one of ${choices.join(', ')}')\n` + } args += ` args.append(str(value))\n` } } else { if (interfaceJsonTypeToInterfaceType.has(parameter.type)) { args += ` if ${snake} is not None:\n` - const interfaceType = interfaceJsonTypeToInterfaceType.get(parameter.type) + const interfaceType = interfaceJsonTypeToInterfaceType.get( + parameter.type + ) if (interfaceType.includes('File')) { // for files args += ` input_file = str(PurePosixPath(${snakeCase(parameter.name)}))\n` @@ -222,6 +233,11 @@ from itkwasm import ( } } else { args += ` if ${snake}:\n` + if (parameter.type.startsWith('TEXT:{')) { + const choices = parameter.type.split('{')[1].split('}')[0].split(',') + args += ` if ${snake} not in (${choices.map((c) => `'${c}'`).join(',')}):\n` + args += ` raise ValueError(f'${snake} must be one of ${choices.join(', ')}')\n` + } args += ` args.append('--${parameter.name}')\n` args += ` args.append(str(${snake}))\n` } @@ -234,23 +250,23 @@ from itkwasm import ( const canonical = canonicalType(type) const pythonType = interfaceJsonTypeToPythonType.get(canonical) switch (pythonType) { - case "os.PathLike": + case 'os.PathLike': return `Path(${value}.data.path)` - case "str": + case 'str': if (type === 'TEXT') { return `${value}` } else { return `${value}.data.data` } - case "bytes": + case 'bytes': return `${value}.data.data` - case "int": + case 'int': return `int(${value})` - case "bool": + case 'bool': return `bool(${value})` - case "float": + case 'float': return `float(${value})` - case "Any": + case 'Any': return `${value}.data` default: return `${value}.data` @@ -258,10 +274,12 @@ from itkwasm import ( } outputCount = 0 const jsonOutputs = interfaceJson['outputs'] - const numOutputs = interfaceJson.outputs.filter(o => !o.type.includes('FILE')).length + const numOutputs = interfaceJson.outputs.filter( + (o) => !o.type.includes('FILE') + ).length if (numOutputs > 1) { postOutput += ' result = (\n' - } else if (numOutputs === 1){ + } else if (numOutputs === 1) { postOutput = ' result = ' } diff --git a/packages/core/typescript/itk-wasm/src/bindgen/typescript/function-module.js b/packages/core/typescript/itk-wasm/src/bindgen/typescript/function-module.js index 316b0f488..4d61c3d6f 100644 --- a/packages/core/typescript/itk-wasm/src/bindgen/typescript/function-module.js +++ b/packages/core/typescript/itk-wasm/src/bindgen/typescript/function-module.js @@ -478,7 +478,13 @@ function functionModule( functionContent += ' args.push(inputCountString)\n\n' } } else { - functionContent += ' args.push(value.toString())\n\n' + if (parameter.type.startsWith('TEXT:{')) { + const choices = parameter.type.split('{')[1].split('}')[0].split(',') + functionContent += ` if (![${choices.map((c) => `'${c}'`).join(', ')}].includes(options.${camel})) {\n` + functionContent += ` throw new Error('"${parameter.name}" option must be one of ${choices.join(', ')}')\n` + functionContent += ' }\n' + } + functionContent += ' args.push(value.toString())\n' } functionContent += forNode ? ' })\n' : ' }))\n' } else { @@ -518,6 +524,12 @@ function functionModule( functionContent += ` args.push('--${parameter.name}', inputCountString)\n\n` } } else { + if (parameter.type.startsWith('TEXT:{')) { + const choices = parameter.type.split('{')[1].split('}')[0].split(',') + functionContent += ` if (![${choices.map((c) => `'${c}'`).join(', ')}].includes(options.${camel})) {\n` + functionContent += ` throw new Error('"${parameter.name}" option must be one of ${choices.join(', ')}')\n` + functionContent += ' }\n' + } functionContent += ` args.push('--${parameter.name}', options.${camel}.toString())\n\n` } } diff --git a/packages/core/typescript/itk-wasm/test/pipelines/input-output-json-pipeline/input-output-json-test.cxx b/packages/core/typescript/itk-wasm/test/pipelines/input-output-json-pipeline/input-output-json-test.cxx index 644b10a68..bee896b6f 100644 --- a/packages/core/typescript/itk-wasm/test/pipelines/input-output-json-pipeline/input-output-json-test.cxx +++ b/packages/core/typescript/itk-wasm/test/pipelines/input-output-json-pipeline/input-output-json-test.cxx @@ -28,6 +28,9 @@ int main( int argc, char * argv[] ) itk::wasm::InputTextStream inputJson; pipeline.add_option("input-json", inputJson, "The input json")->type_name("INPUT_JSON"); + std::string stringChoice = "valuea"; + pipeline.add_option("--string-choice", stringChoice, "A string choice, one of: valuea, valueb, or valuec")->check(CLI::IsMember({"valuea", "valueb", "valuec"})); + itk::wasm::OutputTextStream outputJson; pipeline.add_option("output-json", outputJson, "The output json")->type_name("OUTPUT_JSON"); From ce5fa5d456659fca56f56002033f4334b4b7da37 Mon Sep 17 00:00:00 2001 From: Matt McCormick Date: Tue, 26 Nov 2024 14:38:43 -0500 Subject: [PATCH 2/8] chore(itk-wasm): bump version to 1.0.0-b.183 --- packages/core/typescript/itk-wasm/package.json | 2 +- packages/core/typescript/itk-wasm/src/version.ts | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/core/typescript/itk-wasm/package.json b/packages/core/typescript/itk-wasm/package.json index 566225643..4610e2d58 100644 --- a/packages/core/typescript/itk-wasm/package.json +++ b/packages/core/typescript/itk-wasm/package.json @@ -1,6 +1,6 @@ { "name": "itk-wasm", - "version": "1.0.0-b.182", + "version": "1.0.0-b.183", "description": "High-performance spatial analysis in a web browser, Node.js, and reproducible execution across programming languages and hardware architectures.", "type": "module", "module": "./dist/index.js", diff --git a/packages/core/typescript/itk-wasm/src/version.ts b/packages/core/typescript/itk-wasm/src/version.ts index 94f7e5868..1c318729c 100644 --- a/packages/core/typescript/itk-wasm/src/version.ts +++ b/packages/core/typescript/itk-wasm/src/version.ts @@ -1,3 +1,3 @@ -const version = '1.0.0-b.182' +const version = '1.0.0-b.183' export default version From 1aff94ee1d5f87bd87a9370c18c55cd04820499f Mon Sep 17 00:00:00 2001 From: Matt McCormick Date: Tue, 26 Nov 2024 14:40:18 -0500 Subject: [PATCH 3/8] chore(bindgen): bump itk-wasm to 1.0.0-b.183 --- .../src/bindgen/typescript/resources/template.package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/core/typescript/itk-wasm/src/bindgen/typescript/resources/template.package.json b/packages/core/typescript/itk-wasm/src/bindgen/typescript/resources/template.package.json index a3d973282..77eaec162 100644 --- a/packages/core/typescript/itk-wasm/src/bindgen/typescript/resources/template.package.json +++ b/packages/core/typescript/itk-wasm/src/bindgen/typescript/resources/template.package.json @@ -33,7 +33,7 @@ "author": "", "license": "Apache-2.0", "dependencies": { - "itk-wasm": "1.0.0-b.177" + "itk-wasm": "1.0.0-b.183" }, "devDependencies": { "@itk-wasm/demo-app": "^0.2.0", From 92d15c7d0e0d82b90fa7c378b762ada5be5d4b1e Mon Sep 17 00:00:00 2001 From: Matt McCormick Date: Wed, 27 Nov 2024 12:31:32 -0500 Subject: [PATCH 4/8] fix(bindgen): ident level for wasi string choice param check --- .../itk-wasm/src/bindgen/python/wasi/wasi-function-module.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/core/typescript/itk-wasm/src/bindgen/python/wasi/wasi-function-module.js b/packages/core/typescript/itk-wasm/src/bindgen/python/wasi/wasi-function-module.js index ec3f3f407..6b7bdc83c 100644 --- a/packages/core/typescript/itk-wasm/src/bindgen/python/wasi/wasi-function-module.js +++ b/packages/core/typescript/itk-wasm/src/bindgen/python/wasi/wasi-function-module.js @@ -201,8 +201,8 @@ from itkwasm import ( } else { if (parameter.type.startsWith('TEXT:{')) { const choices = parameter.type.split('{')[1].split('}')[0].split(',') - args += ` if ${snake} not in (${choices.map((c) => `'${c}'`).join(',')}):\n` - args += ` raise ValueError(f'${snake} must be one of ${choices.join(', ')}')\n` + args += ` if ${snake} not in (${choices.map((c) => `'${c}'`).join(',')}):\n` + args += ` raise ValueError(f'${snake} must be one of ${choices.join(', ')}')\n` } args += ` args.append(str(value))\n` } From d2e739f6bb46ff1a6886c6d74075aaddc55488ea Mon Sep 17 00:00:00 2001 From: Matt McCormick Date: Wed, 27 Nov 2024 12:34:06 -0500 Subject: [PATCH 5/8] chore: bump itk-wasm to 1.0.0-b.184 --- packages/core/typescript/itk-wasm/package.json | 2 +- packages/core/typescript/itk-wasm/src/version.ts | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/core/typescript/itk-wasm/package.json b/packages/core/typescript/itk-wasm/package.json index 4610e2d58..de7d30278 100644 --- a/packages/core/typescript/itk-wasm/package.json +++ b/packages/core/typescript/itk-wasm/package.json @@ -1,6 +1,6 @@ { "name": "itk-wasm", - "version": "1.0.0-b.183", + "version": "1.0.0-b.184", "description": "High-performance spatial analysis in a web browser, Node.js, and reproducible execution across programming languages and hardware architectures.", "type": "module", "module": "./dist/index.js", diff --git a/packages/core/typescript/itk-wasm/src/version.ts b/packages/core/typescript/itk-wasm/src/version.ts index 1c318729c..50a2c9489 100644 --- a/packages/core/typescript/itk-wasm/src/version.ts +++ b/packages/core/typescript/itk-wasm/src/version.ts @@ -1,3 +1,3 @@ -const version = '1.0.0-b.183' +const version = '1.0.0-b.184' export default version From d6d78c3f7d229cae81ee465efc4057942e061245 Mon Sep 17 00:00:00 2001 From: Matt McCormick Date: Fri, 6 Dec 2024 13:06:05 -0500 Subject: [PATCH 6/8] ci(python-wasm): bump pyodide-actions/install-browser to latest Plus Windows no-sudo patch. --- .github/workflows/python-wasm.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/python-wasm.yml b/.github/workflows/python-wasm.yml index 39d64e35f..4fe66359b 100644 --- a/.github/workflows/python-wasm.yml +++ b/.github/workflows/python-wasm.yml @@ -71,7 +71,7 @@ jobs: run: | pnpm run --aggregate-output --filter "@itk-wasm/${{ matrix.package }}-build" test:python:wasi - - uses: pyodide/pyodide-actions/install-browser@e3edcb8db4f1ec4604e402013f3ae2ee9f114441 + - uses: thewtex/pyodide-actions/install-browser@1e32e8a037a3e99a845dd7ebad6b057a40b7e2c0 if: ${{ matrix.python-minor-version > 10 }} with: runner: selenium From de8f11bfcf3a0f1fd88f4e1bea0410e7e25b4c05 Mon Sep 17 00:00:00 2001 From: Matt McCormick Date: Mon, 9 Dec 2024 14:01:15 -0500 Subject: [PATCH 7/8] ci(python-wasm): test pyodide with python > 3.11 --- .github/workflows/python-wasm.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/python-wasm.yml b/.github/workflows/python-wasm.yml index 4fe66359b..860fbbde0 100644 --- a/.github/workflows/python-wasm.yml +++ b/.github/workflows/python-wasm.yml @@ -72,13 +72,13 @@ jobs: pnpm run --aggregate-output --filter "@itk-wasm/${{ matrix.package }}-build" test:python:wasi - uses: thewtex/pyodide-actions/install-browser@1e32e8a037a3e99a845dd7ebad6b057a40b7e2c0 - if: ${{ matrix.python-minor-version > 10 }} + if: ${{ matrix.python-minor-version > 11 }} with: runner: selenium browser: chrome browser-version: latest - name: Test python on chrome - if: ${{ matrix.python-minor-version > 10 && matrix.package != 'dicom' && matrix.os == 'ubuntu-22.04' || (matrix.package != 'mesh-io' && matrix.package != 'image-io' && matrix.package != 'dicom' && matrix.os != 'macos-14' )}} + if: ${{ matrix.python-minor-version > 11 && matrix.package != 'dicom' && matrix.os == 'ubuntu-22.04' || (matrix.package != 'mesh-io' && matrix.package != 'image-io' && matrix.package != 'dicom' && matrix.os != 'macos-14' )}} run: | pnpm run --aggregate-output --filter "@itk-wasm/${{ matrix.package }}-build" test:python:emscripten From 81090e0f534d458a35fa569629e9f0b4b38628a8 Mon Sep 17 00:00:00 2001 From: Matt McCormick Date: Mon, 9 Dec 2024 14:57:52 -0500 Subject: [PATCH 8/8] ci(python-wasm): only test pyodide on ubuntu-22.04 This is currently an incompatibility between the selenium and Chrome versions on Windows: ``` ----------------------------- Captured log setup ------------------------------ WARNING selenium.webdriver.common.selenium_manager:selenium_manager.py:138 The chromedriver version (133.0.6888.0) detected in PATH at C:\hostedtoolcache\windows\setup-chrome\chromedriver\1393741\x64\chromedriver.exe might not be compatible with the detected chrome version (131.0.6778.86); currently, chromedriver 131.0.6778.87 is recommended for chrome 131.*, so it is advised to delete the driver in PATH and retry =========================== short test summary info =========================== ERROR test/test_compress_stringify.py::test_example[chrome] - selenium.common.exceptions.SessionNotCreatedException: Message: session not created: This version of ChromeDriver only supports Chrome version 133 Current browser version is 131.0.6778.86 with binary path C:\Program Files\Google\Chrome\Application\chrome.exe Stacktrace: GetHandleVerifier [0x00007FF66A01C635+53269] simdutf::get_active_implementation [0x00007FF669E5D3B7+1684007] (No symbol) [0x00007FF669AFE09B] (No symbol) [0x00007FF669B43DB9] (No symbol) [0x00007FF669B42C00] (No symbol) [0x00007FF669B3D79D] ``` --- .github/workflows/python-wasm.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/python-wasm.yml b/.github/workflows/python-wasm.yml index 860fbbde0..f1bea7929 100644 --- a/.github/workflows/python-wasm.yml +++ b/.github/workflows/python-wasm.yml @@ -72,13 +72,13 @@ jobs: pnpm run --aggregate-output --filter "@itk-wasm/${{ matrix.package }}-build" test:python:wasi - uses: thewtex/pyodide-actions/install-browser@1e32e8a037a3e99a845dd7ebad6b057a40b7e2c0 - if: ${{ matrix.python-minor-version > 11 }} + if: ${{ matrix.python-minor-version > 11 && matrix.os == 'ubuntu-22.04' }} with: runner: selenium browser: chrome browser-version: latest - name: Test python on chrome - if: ${{ matrix.python-minor-version > 11 && matrix.package != 'dicom' && matrix.os == 'ubuntu-22.04' || (matrix.package != 'mesh-io' && matrix.package != 'image-io' && matrix.package != 'dicom' && matrix.os != 'macos-14' )}} + if: ${{ matrix.python-minor-version > 11 && matrix.package != 'dicom' && matrix.os == 'ubuntu-22.04' }} run: | pnpm run --aggregate-output --filter "@itk-wasm/${{ matrix.package }}-build" test:python:emscripten