From 27979597bb24c07d9616fd78401c6e5d55a62b2e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Isaac=20Rold=C3=A1n?= Date: Mon, 18 Nov 2024 13:12:24 +0100 Subject: [PATCH 1/2] add tests for dev-session errors --- .../dev/processes/dev-session.test.ts | 159 ++++++++++++++++++ 1 file changed, 159 insertions(+) create mode 100644 packages/app/src/cli/services/dev/processes/dev-session.test.ts diff --git a/packages/app/src/cli/services/dev/processes/dev-session.test.ts b/packages/app/src/cli/services/dev/processes/dev-session.test.ts new file mode 100644 index 00000000000..0e91ed04a00 --- /dev/null +++ b/packages/app/src/cli/services/dev/processes/dev-session.test.ts @@ -0,0 +1,159 @@ +import {setupDevSessionProcess, pushUpdatesForDevSession} from './dev-session.js' +import {DeveloperPlatformClient} from '../../../utilities/developer-platform-client.js' +import {AppLinkedInterface} from '../../../models/app/app.js' +import {AppEventWatcher} from '../app-events/app-event-watcher.js' +import {buildAppURLForWeb} from '../../../utilities/app/app-url.js' +import {testAppLinked, testDeveloperPlatformClient, testWebhookExtensions} from '../../../models/app/app.test-data.js' +import {formData} from '@shopify/cli-kit/node/http' +import {describe, expect, test, vi, beforeEach} from 'vitest' +import {AbortSignal} from '@shopify/cli-kit/node/abort' +import {flushPromises} from '@shopify/cli-kit/node/promises' + +vi.mock('@shopify/cli-kit/node/fs') +vi.mock('@shopify/cli-kit/node/archiver') +vi.mock('@shopify/cli-kit/node/http') +vi.mock('../../../utilities/app/app-url.js') +vi.mock('node-fetch') + +vi.mocked(formData).mockReturnValue({append: vi.fn(), getHeaders: vi.fn()} as any) + +describe('setupDevSessionProcess', () => { + test('returns a dev session process with correct configuration', async () => { + // Given + const options = { + app: {} as AppLinkedInterface, + apiKey: 'test-api-key', + developerPlatformClient: {} as DeveloperPlatformClient, + storeFqdn: 'test.myshopify.com', + url: 'https://test.dev', + organizationId: 'org123', + appId: 'app123', + appWatcher: {} as AppEventWatcher, + } + + // When + const process = await setupDevSessionProcess(options) + + // Then + expect(process).toEqual({ + type: 'dev-session', + prefix: 'dev-session', + function: pushUpdatesForDevSession, + options: { + app: options.app, + apiKey: options.apiKey, + developerPlatformClient: options.developerPlatformClient, + storeFqdn: options.storeFqdn, + url: options.url, + organizationId: options.organizationId, + appId: options.appId, + appWatcher: options.appWatcher, + }, + }) + }) +}) + +describe('pushUpdatesForDevSession', () => { + let stdout: any + let stderr: any + let options: any + let developerPlatformClient: any + let appWatcher: AppEventWatcher + let app: AppLinkedInterface + + beforeEach(() => { + vi.mocked(formData).mockReturnValue({append: vi.fn(), getHeaders: vi.fn()} as any) + stdout = {write: vi.fn()} + stderr = {write: vi.fn()} + developerPlatformClient = testDeveloperPlatformClient() + app = testAppLinked() + appWatcher = new AppEventWatcher(app) + + options = { + developerPlatformClient, + appWatcher, + storeFqdn: 'test.myshopify.com', + appId: 'app123', + organizationId: 'org123', + } + }) + + test('creates a new dev session successfully when receiving the app watcher start event', async () => { + // When + await pushUpdatesForDevSession({stderr, stdout, abortSignal: new AbortSignal()}, options) + await appWatcher.start() + await flushPromises() + + // Then + expect(developerPlatformClient.devSessionCreate).toHaveBeenCalled() + expect(stdout.write).toHaveBeenCalledWith(expect.stringContaining('Ready')) + }) + + test('handles user errors from dev session creation', async () => { + // Given + const userErrors = [{message: 'Test error', category: 'test'}] + developerPlatformClient.devSessionCreate = vi.fn().mockResolvedValue({devSessionCreate: {userErrors}}) + const exitSpy = vi.spyOn(process, 'exit') + + // When + await pushUpdatesForDevSession({stderr, stdout, abortSignal: new AbortSignal()}, options) + await appWatcher.start() + await flushPromises() + + // Then + expect(stdout.write).toHaveBeenCalledWith(expect.stringContaining('Error')) + expect(stdout.write).toHaveBeenCalledWith(expect.stringContaining('Test error')) + expect(exitSpy).toHaveBeenCalledWith(1) + }) + + test('handles receiving an event before session is ready', async () => { + // When + await pushUpdatesForDevSession({stderr, stdout, abortSignal: new AbortSignal()}, options) + appWatcher.emit('all', {app, extensionEvents: [{type: 'updated', extension: testWebhookExtensions()}]}) + await flushPromises() + + // Then + expect(stdout.write).toHaveBeenCalledWith( + expect.stringContaining('Change detected, but dev session is not ready yet.'), + ) + expect(developerPlatformClient.devSessionCreate).not.toHaveBeenCalled() + expect(developerPlatformClient.devSessionUpdate).not.toHaveBeenCalled() + }) + + test('handles user errors from dev session update', async () => { + // Given + const userErrors = [{message: 'Update error', category: 'test'}] + developerPlatformClient.devSessionUpdate = vi.fn().mockResolvedValue({devSessionUpdate: {userErrors}}) + const exitSpy = vi.spyOn(process, 'exit') + + // When + await pushUpdatesForDevSession({stderr, stdout, abortSignal: new AbortSignal()}, options) + await appWatcher.start() + await flushPromises() + appWatcher.emit('all', {app, extensionEvents: [{type: 'updated', extension: testWebhookExtensions()}]}) + await flushPromises() + + // Then + expect(stdout.write).toHaveBeenCalledWith(expect.stringContaining('Error')) + expect(stdout.write).toHaveBeenCalledWith(expect.stringContaining('Update error')) + expect(exitSpy).not.toHaveBeenCalled() + }) + + test('handles scope changes and displays action required message', async () => { + // Given + vi.mocked(buildAppURLForWeb).mockResolvedValue('https://test.myshopify.com/admin/apps/test') + const event = {extensionEvents: [{type: 'updated', extension: {handle: 'app-access'}}], app} + + // When + await pushUpdatesForDevSession({stderr, stdout, abortSignal: new AbortSignal()}, options) + await appWatcher.start() + await flushPromises() + appWatcher.emit('all', event) + await flushPromises() + + // Then + expect(stdout.write).toHaveBeenCalledWith(expect.stringContaining('Updated')) + expect(stdout.write).toHaveBeenCalledWith(expect.stringContaining('Action required')) + expect(stdout.write).toHaveBeenCalledWith(expect.stringContaining('Scopes updated')) + }) +}) From 2d13c1b7de097b0cac946314e30cf68c080c27ea Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Isaac=20Rold=C3=A1n?= Date: Mon, 18 Nov 2024 15:42:30 +0100 Subject: [PATCH 2/2] Fix tests --- .../dev/processes/dev-session.test.ts | 41 +++++++++++++++---- .../cli/services/dev/processes/dev-session.ts | 7 +++- 2 files changed, 38 insertions(+), 10 deletions(-) diff --git a/packages/app/src/cli/services/dev/processes/dev-session.test.ts b/packages/app/src/cli/services/dev/processes/dev-session.test.ts index 0e91ed04a00..3a4a6a9bed2 100644 --- a/packages/app/src/cli/services/dev/processes/dev-session.test.ts +++ b/packages/app/src/cli/services/dev/processes/dev-session.test.ts @@ -3,11 +3,18 @@ import {DeveloperPlatformClient} from '../../../utilities/developer-platform-cli import {AppLinkedInterface} from '../../../models/app/app.js' import {AppEventWatcher} from '../app-events/app-event-watcher.js' import {buildAppURLForWeb} from '../../../utilities/app/app-url.js' -import {testAppLinked, testDeveloperPlatformClient, testWebhookExtensions} from '../../../models/app/app.test-data.js' +import { + testAppLinked, + testDeveloperPlatformClient, + testUIExtension, + testWebhookExtensions, +} from '../../../models/app/app.test-data.js' import {formData} from '@shopify/cli-kit/node/http' import {describe, expect, test, vi, beforeEach} from 'vitest' import {AbortSignal} from '@shopify/cli-kit/node/abort' import {flushPromises} from '@shopify/cli-kit/node/promises' +import {writeFile} from '@shopify/cli-kit/node/fs' +import * as outputContext from '@shopify/cli-kit/node/ui/components' vi.mock('@shopify/cli-kit/node/fs') vi.mock('@shopify/cli-kit/node/archiver') @@ -15,8 +22,6 @@ vi.mock('@shopify/cli-kit/node/http') vi.mock('../../../utilities/app/app-url.js') vi.mock('node-fetch') -vi.mocked(formData).mockReturnValue({append: vi.fn(), getHeaders: vi.fn()} as any) - describe('setupDevSessionProcess', () => { test('returns a dev session process with correct configuration', async () => { // Given @@ -63,6 +68,7 @@ describe('pushUpdatesForDevSession', () => { beforeEach(() => { vi.mocked(formData).mockReturnValue({append: vi.fn(), getHeaders: vi.fn()} as any) + vi.mocked(writeFile).mockResolvedValue(undefined) stdout = {write: vi.fn()} stderr = {write: vi.fn()} developerPlatformClient = testDeveloperPlatformClient() @@ -89,21 +95,40 @@ describe('pushUpdatesForDevSession', () => { expect(stdout.write).toHaveBeenCalledWith(expect.stringContaining('Ready')) }) + test('updates use the extension handle as the output prefix', async () => { + // When + + const spyContext = vi.spyOn(outputContext, 'useConcurrentOutputContext') + await pushUpdatesForDevSession({stderr, stdout, abortSignal: new AbortSignal()}, options) + await appWatcher.start() + await flushPromises() + + const extension = await testUIExtension() + appWatcher.emit('all', {app, extensionEvents: [{type: 'updated', extension}]}) + await flushPromises() + + // Then + expect(stdout.write).toHaveBeenCalledWith(expect.stringContaining('Updated')) + expect(spyContext).toHaveBeenCalledWith({outputPrefix: 'test-ui-extension', stripAnsi: false}, expect.anything()) + + // In theory this shouldn't be necessary, but vitest doesn't restore spies automatically. + // eslint-disable-next-line @shopify/cli/no-vi-manual-mock-clear + vi.restoreAllMocks() + }) + test('handles user errors from dev session creation', async () => { // Given const userErrors = [{message: 'Test error', category: 'test'}] developerPlatformClient.devSessionCreate = vi.fn().mockResolvedValue({devSessionCreate: {userErrors}}) - const exitSpy = vi.spyOn(process, 'exit') // When - await pushUpdatesForDevSession({stderr, stdout, abortSignal: new AbortSignal()}, options) await appWatcher.start() + await pushUpdatesForDevSession({stderr, stdout, abortSignal: new AbortSignal()}, options) await flushPromises() // Then expect(stdout.write).toHaveBeenCalledWith(expect.stringContaining('Error')) expect(stdout.write).toHaveBeenCalledWith(expect.stringContaining('Test error')) - expect(exitSpy).toHaveBeenCalledWith(1) }) test('handles receiving an event before session is ready', async () => { @@ -124,7 +149,6 @@ describe('pushUpdatesForDevSession', () => { // Given const userErrors = [{message: 'Update error', category: 'test'}] developerPlatformClient.devSessionUpdate = vi.fn().mockResolvedValue({devSessionUpdate: {userErrors}}) - const exitSpy = vi.spyOn(process, 'exit') // When await pushUpdatesForDevSession({stderr, stdout, abortSignal: new AbortSignal()}, options) @@ -136,13 +160,13 @@ describe('pushUpdatesForDevSession', () => { // Then expect(stdout.write).toHaveBeenCalledWith(expect.stringContaining('Error')) expect(stdout.write).toHaveBeenCalledWith(expect.stringContaining('Update error')) - expect(exitSpy).not.toHaveBeenCalled() }) test('handles scope changes and displays action required message', async () => { // Given vi.mocked(buildAppURLForWeb).mockResolvedValue('https://test.myshopify.com/admin/apps/test') const event = {extensionEvents: [{type: 'updated', extension: {handle: 'app-access'}}], app} + const contextSpy = vi.spyOn(outputContext, 'useConcurrentOutputContext') // When await pushUpdatesForDevSession({stderr, stdout, abortSignal: new AbortSignal()}, options) @@ -155,5 +179,6 @@ describe('pushUpdatesForDevSession', () => { expect(stdout.write).toHaveBeenCalledWith(expect.stringContaining('Updated')) expect(stdout.write).toHaveBeenCalledWith(expect.stringContaining('Action required')) expect(stdout.write).toHaveBeenCalledWith(expect.stringContaining('Scopes updated')) + expect(contextSpy).toHaveBeenCalledWith({outputPrefix: 'dev-session', stripAnsi: false}, expect.anything()) }) }) diff --git a/packages/app/src/cli/services/dev/processes/dev-session.ts b/packages/app/src/cli/services/dev/processes/dev-session.ts index ae29a67ea3d..df682b62a0d 100644 --- a/packages/app/src/cli/services/dev/processes/dev-session.ts +++ b/packages/app/src/cli/services/dev/processes/dev-session.ts @@ -15,6 +15,7 @@ import {performActionWithRetryAfterRecovery} from '@shopify/cli-kit/common/retry import {useConcurrentOutputContext} from '@shopify/cli-kit/node/ui/components' import {JsonMapType} from '@shopify/cli-kit/node/toml' import {AbortError} from '@shopify/cli-kit/node/error' +import {isUnitTest} from '@shopify/cli-kit/node/context/local' import {Writable} from 'stream' interface DevSessionOptions { @@ -83,6 +84,7 @@ export const pushUpdatesForDevSession: DevProcessFunction = a ) => { const {developerPlatformClient, appWatcher} = options + isDevSessionReady = false const refreshToken = async () => { return developerPlatformClient.refreshToken() } @@ -155,8 +157,9 @@ async function handleDevSessionResult( await processUserErrors(result.error, processOptions, processOptions.stdout) } - // If we failed to create a session, exit the process - if (!isDevSessionReady) throw new AbortError('Failed to create dev session') + // If we failed to create a session, exit the process. Don't throw an error in tests as it can't be caught due to the + // async nature of the process. + if (!isDevSessionReady && !isUnitTest()) throw new AbortError('Failed to create dev session') } /**