From 1c9d74988e2dad10d1753db684b29d938de1a2e2 Mon Sep 17 00:00:00 2001 From: Jeff Charles Date: Mon, 11 Nov 2024 15:12:33 -0500 Subject: [PATCH 1/6] Download and use Javy plugin --- .changeset/dry-rice-guess.md | 5 ++ .../cli/services/function/binaries.test.ts | 41 ++++++++++++-- .../app/src/cli/services/function/binaries.ts | 56 ++++++++++++++++--- .../src/cli/services/function/build.test.ts | 15 ++++- .../app/src/cli/services/function/build.ts | 22 ++++++-- .../src/cli/services/function/runner.test.ts | 6 +- .../app/src/cli/services/function/runner.ts | 4 +- 7 files changed, 124 insertions(+), 25 deletions(-) create mode 100644 .changeset/dry-rice-guess.md diff --git a/.changeset/dry-rice-guess.md b/.changeset/dry-rice-guess.md new file mode 100644 index 00000000000..7002377b8b4 --- /dev/null +++ b/.changeset/dry-rice-guess.md @@ -0,0 +1,5 @@ +--- +'@shopify/app': patch +--- + +Update Javy invocation to use Javy plugin diff --git a/packages/app/src/cli/services/function/binaries.test.ts b/packages/app/src/cli/services/function/binaries.test.ts index c354a69a80c..7656cd9e2f0 100644 --- a/packages/app/src/cli/services/function/binaries.test.ts +++ b/packages/app/src/cli/services/function/binaries.test.ts @@ -1,10 +1,11 @@ -import {javyBinary, functionRunnerBinary, installBinary} from './binaries.js' +import {javyBinary, functionRunnerBinary, downloadBinary, javyPluginBinary} from './binaries.js' import {fetch, Response} from '@shopify/cli-kit/node/http' import {fileExists, removeFile} from '@shopify/cli-kit/node/fs' import {describe, expect, test, vi} from 'vitest' import {gzipSync} from 'zlib' const javy = javyBinary() +const javyPlugin = javyPluginBinary() const functionRunner = functionRunnerBinary() vi.mock('@shopify/cli-kit/node/http', async () => { @@ -117,14 +118,14 @@ describe('javy', () => { }) }) - test('installs Javy', async () => { + test('downloads Javy', async () => { // Given await removeFile(javy.path) await expect(fileExists(javy.path)).resolves.toBeFalsy() vi.mocked(fetch).mockResolvedValue(new Response(gzipSync('javy binary'))) // When - await installBinary(javy) + await downloadBinary(javy) // Then expect(fetch).toHaveBeenCalledOnce() @@ -132,6 +133,36 @@ describe('javy', () => { }) }) +describe('javy-plugin', () => { + test('properties are set correctly', () => { + expect(javyPlugin.name).toBe('javy_quickjs_provider_v3') + expect(javyPlugin.version).match(/^v\d.\d.\d$/) + expect(javyPlugin.path).toMatch(/(\/|\\)javy_quickjs_provider_v3.wasm$/) + }) + + test('downloadUrl returns the correct URL', () => { + // When + const url = javyPlugin.downloadUrl('', '') + + // Then + expect(url).toMatch(/https:\/\/github.com\/bytecodealliance\/javy\/releases\/download\/v\d\.\d\.\d\/plugin.wasm.gz/) + }) + + test('downloads javy-plugin', async () => { + // Given + await removeFile(javyPlugin.path) + await expect(fileExists(javyPlugin.path)).resolves.toBeFalsy() + vi.mocked(fetch).mockResolvedValue(new Response(gzipSync('javy-plugin binary'))) + + // When + await downloadBinary(javyPlugin) + + // Then + expect(fetch).toHaveBeenCalledOnce() + await expect(fileExists(javyPlugin.path)).resolves.toBeTruthy() + }) +}) + describe('functionRunner', () => { test('properties are set correctly', () => { expect(functionRunner.name).toBe('function-runner') @@ -234,14 +265,14 @@ describe('functionRunner', () => { }) }) - test('installs function-runner', async () => { + test('downloads function-runner', async () => { // Given await removeFile(functionRunner.path) await expect(fileExists(functionRunner.path)).resolves.toBeFalsy() vi.mocked(fetch).mockResolvedValue(new Response(gzipSync('function-runner binary'))) // When - await installBinary(functionRunner) + await downloadBinary(functionRunner) // Then expect(fetch).toHaveBeenCalledOnce() diff --git a/packages/app/src/cli/services/function/binaries.ts b/packages/app/src/cli/services/function/binaries.ts index 7e7521af6ad..422ac24b032 100644 --- a/packages/app/src/cli/services/function/binaries.ts +++ b/packages/app/src/cli/services/function/binaries.ts @@ -11,11 +11,21 @@ import {fileURLToPath} from 'node:url' const FUNCTION_RUNNER_VERSION = 'v6.3.0' const JAVY_VERSION = 'v3.2.0' +const JAVY_PLUGIN_VERSION = 'v3.2.0' + +interface DownloadableBinary { + path: string + name: string + version: string + + downloadUrl(processPlatform: string, processArch: string): string + processResponse(responseStream: PipelineSource, outputStream: fs.WriteStream): Promise +} // The logic for determining the download URL and what to do with the response stream is _coincidentally_ the same for // Javy and function-runner for now. Those methods may not continue to have the same logic in the future. If they -// diverge, make `Binary` an abstract class and create subclasses to handle the different logic polymorphically. -class DownloadableBinary { +// diverge, create different classes to handle the different logic polymorphically. +class Executable implements DownloadableBinary { readonly name: string readonly version: string readonly path: string @@ -71,29 +81,56 @@ class DownloadableBinary { } async processResponse(responseStream: PipelineSource, outputStream: fs.WriteStream): Promise { - const gunzip = gzip.createGunzip() - await stream.pipeline(responseStream, gunzip, outputStream) + return gunzipResponse(responseStream, outputStream) + } +} + +class JavyPlugin implements DownloadableBinary { + readonly name: string + readonly version: string + readonly path: string + + constructor() { + this.name = 'javy_quickjs_provider_v3' + this.version = JAVY_PLUGIN_VERSION + this.path = joinPath(dirname(fileURLToPath(import.meta.url)), '..', 'bin', 'javy_quickjs_provider_v3.wasm') + } + + downloadUrl(_processPlatform: string, _processArch: string) { + return `https://github.com/bytecodealliance/javy/releases/download/${this.version}/plugin.wasm.gz` + } + + async processResponse(responseStream: PipelineSource, outputStream: fs.WriteStream): Promise { + return gunzipResponse(responseStream, outputStream) } } let _javy: DownloadableBinary +let _javyPlugin: DownloadableBinary let _functionRunner: DownloadableBinary export function javyBinary() { if (!_javy) { - _javy = new DownloadableBinary('javy', JAVY_VERSION, 'bytecodealliance/javy') + _javy = new Executable('javy', JAVY_VERSION, 'bytecodealliance/javy') } return _javy } +export function javyPluginBinary() { + if (!_javyPlugin) { + _javyPlugin = new JavyPlugin() + } + return _javyPlugin +} + export function functionRunnerBinary() { if (!_functionRunner) { - _functionRunner = new DownloadableBinary('function-runner', FUNCTION_RUNNER_VERSION, 'Shopify/function-runner') + _functionRunner = new Executable('function-runner', FUNCTION_RUNNER_VERSION, 'Shopify/function-runner') } return _functionRunner } -export async function installBinary(bin: DownloadableBinary) { +export async function downloadBinary(bin: DownloadableBinary) { const isInstalled = await fileExists(bin.path) if (isInstalled) { return @@ -132,3 +169,8 @@ export async function installBinary(bin: DownloadableBinary) { 2, ) } + +async function gunzipResponse(responseStream: PipelineSource, outputStream: fs.WriteStream): Promise { + const gunzip = gzip.createGunzip() + await stream.pipeline(responseStream, gunzip, outputStream) +} diff --git a/packages/app/src/cli/services/function/build.test.ts b/packages/app/src/cli/services/function/build.test.ts index 583b0607247..b1ca9f3f7ea 100644 --- a/packages/app/src/cli/services/function/build.test.ts +++ b/packages/app/src/cli/services/function/build.test.ts @@ -1,5 +1,5 @@ import {buildGraphqlTypes, bundleExtension, runJavy, ExportJavyBuilder, jsExports} from './build.js' -import {javyBinary} from './binaries.js' +import {javyBinary, javyPluginBinary} from './binaries.js' import {testApp, testFunctionExtension} from '../../models/app/app.test-data.js' import {beforeEach, describe, expect, test, vi} from 'vitest' import {exec} from '@shopify/cli-kit/node/system' @@ -143,7 +143,16 @@ describe('runJavy', () => { await expect(got).resolves.toBeUndefined() expect(exec).toHaveBeenCalledWith( javyBinary().path, - ['build', '-C', 'dynamic', '-o', joinPath(ourFunction.directory, 'dist/index.wasm'), 'dist/function.js'], + [ + 'build', + '-C', + 'dynamic', + '-C', + `plugin=${javyPluginBinary().path}`, + '-o', + joinPath(ourFunction.directory, 'dist/index.wasm'), + 'dist/function.js', + ], { cwd: ourFunction.directory, stderr: 'inherit', @@ -230,6 +239,8 @@ describe('ExportJavyBuilder', () => { '-C', 'dynamic', '-C', + `plugin=${javyPluginBinary().path}`, + '-C', expect.stringContaining('wit='), '-C', 'wit-world=shopify-function', diff --git a/packages/app/src/cli/services/function/build.ts b/packages/app/src/cli/services/function/build.ts index 5cd2e9dfa81..101a324404d 100644 --- a/packages/app/src/cli/services/function/build.ts +++ b/packages/app/src/cli/services/function/build.ts @@ -1,5 +1,4 @@ -/* eslint-disable @typescript-eslint/no-non-null-assertion */ -import {installBinary, javyBinary} from './binaries.js' +import {downloadBinary, javyBinary, javyPluginBinary} from './binaries.js' import {ExtensionInstance} from '../../models/extensions/extension-instance.js' import {FunctionConfigType} from '../../models/extensions/specifications/function.js' import {AppInterface} from '../../models/app/app.js' @@ -168,12 +167,23 @@ export async function runJavy( extra: string[] = [], ) { const javy = javyBinary() - await installBinary(javy) + const plugin = javyPluginBinary() + await Promise.all([downloadBinary(javy), downloadBinary(plugin)]) // Using the `build` command we want to emit: // - // `javy build -C dynamic -C wit= -C wit-world=val -o ` - const args = ['build', '-C', 'dynamic', ...extra, '-o', fun.outputPath, 'dist/function.js'] + // `javy build -C dynamic -C plugin=path/to/javy_quickjs_provider_v3.wasm -C wit= -C wit-world=val -o ` + const args = [ + 'build', + '-C', + 'dynamic', + '-C', + `plugin=${plugin.path}`, + ...extra, + '-o', + fun.outputPath, + 'dist/function.js', + ] return exec(javy.path, args, { cwd: fun.directory, @@ -187,7 +197,7 @@ export async function installJavy(app: AppInterface) { const javyRequired = app.allExtensions.some((ext) => ext.features.includes('function') && ext.isJavaScript) if (javyRequired) { const javy = javyBinary() - await installBinary(javy) + await downloadBinary(javy) } } diff --git a/packages/app/src/cli/services/function/runner.test.ts b/packages/app/src/cli/services/function/runner.test.ts index c84cd2202b6..2e78d04d545 100644 --- a/packages/app/src/cli/services/function/runner.test.ts +++ b/packages/app/src/cli/services/function/runner.test.ts @@ -1,5 +1,5 @@ import {runFunction} from './runner.js' -import {functionRunnerBinary, installBinary} from './binaries.js' +import {functionRunnerBinary, downloadBinary} from './binaries.js' import {testFunctionExtension} from '../../models/app/app.test-data.js' import {describe, test, vi, expect} from 'vitest' import {exec} from '@shopify/cli-kit/node/system' @@ -10,7 +10,7 @@ vi.mock('./binaries.js', async (importOriginal) => { const original = await importOriginal() return { ...original, - installBinary: vi.fn().mockResolvedValue(undefined), + downloadBinary: vi.fn().mockResolvedValue(undefined), } }) @@ -23,7 +23,7 @@ describe('runFunction', () => { await runFunction({functionExtension}) // Then - expect(installBinary).toHaveBeenCalledOnce() + expect(downloadBinary).toHaveBeenCalledOnce() }) test('runs function with options', async () => { diff --git a/packages/app/src/cli/services/function/runner.ts b/packages/app/src/cli/services/function/runner.ts index 901a0f546db..9ec0e48169b 100644 --- a/packages/app/src/cli/services/function/runner.ts +++ b/packages/app/src/cli/services/function/runner.ts @@ -1,4 +1,4 @@ -import {functionRunnerBinary, installBinary} from './binaries.js' +import {functionRunnerBinary, downloadBinary} from './binaries.js' import {ExtensionInstance} from '../../models/extensions/extension-instance.js' import {FunctionConfigType} from '../../models/extensions/specifications/function.js' import {exec} from '@shopify/cli-kit/node/system' @@ -19,7 +19,7 @@ interface FunctionRunnerOptions { export async function runFunction(options: FunctionRunnerOptions) { const functionRunner = functionRunnerBinary() - await installBinary(functionRunner) + await downloadBinary(functionRunner) const args: string[] = [] if (options.inputPath) { From ef5ac66b4c63d1f9b1a4988cfa7f0cd70f680633 Mon Sep 17 00:00:00 2001 From: Jeff Charles Date: Mon, 11 Nov 2024 16:13:07 -0500 Subject: [PATCH 2/6] Restore eslint-disable the formatter removed --- packages/app/src/cli/services/function/build.ts | 1 + 1 file changed, 1 insertion(+) diff --git a/packages/app/src/cli/services/function/build.ts b/packages/app/src/cli/services/function/build.ts index 101a324404d..59179aeaca7 100644 --- a/packages/app/src/cli/services/function/build.ts +++ b/packages/app/src/cli/services/function/build.ts @@ -1,3 +1,4 @@ +/* eslint-disable @typescript-eslint/no-non-null-assertion */ import {downloadBinary, javyBinary, javyPluginBinary} from './binaries.js' import {ExtensionInstance} from '../../models/extensions/extension-instance.js' import {FunctionConfigType} from '../../models/extensions/specifications/function.js' From 1f20e460ea6c2b71ce6a21dd705cc1f02774cd1c Mon Sep 17 00:00:00 2001 From: Jeff Charles Date: Wed, 20 Nov 2024 10:32:35 -0500 Subject: [PATCH 3/6] Update to Javy v4.0.0 --- packages/app/src/cli/services/function/binaries.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/app/src/cli/services/function/binaries.ts b/packages/app/src/cli/services/function/binaries.ts index 422ac24b032..2680ac863a1 100644 --- a/packages/app/src/cli/services/function/binaries.ts +++ b/packages/app/src/cli/services/function/binaries.ts @@ -10,8 +10,8 @@ import * as gzip from 'node:zlib' import {fileURLToPath} from 'node:url' const FUNCTION_RUNNER_VERSION = 'v6.3.0' -const JAVY_VERSION = 'v3.2.0' -const JAVY_PLUGIN_VERSION = 'v3.2.0' +const JAVY_VERSION = 'v4.0.0' +const JAVY_PLUGIN_VERSION = 'v4.0.0' interface DownloadableBinary { path: string From 6d0fa5c404beb44c660ca28050f796adc1c2ecab Mon Sep 17 00:00:00 2001 From: Jeff Charles Date: Fri, 22 Nov 2024 09:52:27 -0500 Subject: [PATCH 4/6] `isInstalled` -> `isDownloaded` --- packages/app/src/cli/services/function/binaries.ts | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/packages/app/src/cli/services/function/binaries.ts b/packages/app/src/cli/services/function/binaries.ts index 2680ac863a1..0c487290516 100644 --- a/packages/app/src/cli/services/function/binaries.ts +++ b/packages/app/src/cli/services/function/binaries.ts @@ -131,8 +131,8 @@ export function functionRunnerBinary() { } export async function downloadBinary(bin: DownloadableBinary) { - const isInstalled = await fileExists(bin.path) - if (isInstalled) { + const isDownloaded = await fileExists(bin.path) + if (isDownloaded) { return } @@ -155,7 +155,7 @@ export async function downloadBinary(bin: DownloadableBinary) { } // Download to a temp location and then move the file only after it's fully processed - // so the `isInstalled` check above will continue to return false if the file hasn't + // so the `isDownloaded` check above will continue to return false if the file hasn't // been fully processed. await inTemporaryDirectory(async (tmpDir) => { const tmpFile = joinPath(tmpDir, 'binary') From 96a5856e7df1648947877a1378dbe3f796ddb111 Mon Sep 17 00:00:00 2001 From: Jeff Charles Date: Fri, 22 Nov 2024 09:54:29 -0500 Subject: [PATCH 5/6] Add comment about plugin version --- packages/app/src/cli/services/function/binaries.ts | 2 ++ 1 file changed, 2 insertions(+) diff --git a/packages/app/src/cli/services/function/binaries.ts b/packages/app/src/cli/services/function/binaries.ts index 0c487290516..35984cd81f7 100644 --- a/packages/app/src/cli/services/function/binaries.ts +++ b/packages/app/src/cli/services/function/binaries.ts @@ -11,6 +11,8 @@ import {fileURLToPath} from 'node:url' const FUNCTION_RUNNER_VERSION = 'v6.3.0' const JAVY_VERSION = 'v4.0.0' +// The Javy plugin version does not need to match the Javy version. It should +// match the plugin version used in the function-runner version specified above. const JAVY_PLUGIN_VERSION = 'v4.0.0' interface DownloadableBinary { From fa2656eec0fb11cb039fcf3e330e8942b731090b Mon Sep 17 00:00:00 2001 From: Jeff Charles Date: Fri, 22 Nov 2024 10:02:02 -0500 Subject: [PATCH 6/6] Use plugin version that matches function-runner's --- packages/app/src/cli/services/function/binaries.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/app/src/cli/services/function/binaries.ts b/packages/app/src/cli/services/function/binaries.ts index 35984cd81f7..034d28e18f3 100644 --- a/packages/app/src/cli/services/function/binaries.ts +++ b/packages/app/src/cli/services/function/binaries.ts @@ -13,7 +13,7 @@ const FUNCTION_RUNNER_VERSION = 'v6.3.0' const JAVY_VERSION = 'v4.0.0' // The Javy plugin version does not need to match the Javy version. It should // match the plugin version used in the function-runner version specified above. -const JAVY_PLUGIN_VERSION = 'v4.0.0' +const JAVY_PLUGIN_VERSION = 'v3.2.0' interface DownloadableBinary { path: string