Skip to content

Commit

Permalink
Refactor FileWatcher into a Class
Browse files Browse the repository at this point in the history
  • Loading branch information
isaacroldan committed Nov 25, 2024
1 parent 50dba6a commit 1b4002b
Show file tree
Hide file tree
Showing 7 changed files with 177 additions and 160 deletions.
2 changes: 0 additions & 2 deletions packages/app/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -67,7 +67,6 @@
"graphql-request": "5.2.0",
"h3": "0.7.21",
"http-proxy": "1.18.1",
"micromatch": "4.0.8",
"proper-lockfile": "4.1.2",
"react": "^18.2.0",
"react-dom": "18.2.0",
Expand All @@ -78,7 +77,6 @@
"@types/diff": "^5.0.3",
"@types/express": "^4.17.17",
"@types/http-proxy": "^1.17.11",
"@types/micromatch": "^4.0.8",
"@types/node": "18.19.3",
"@types/proper-lockfile": "4.1.4",
"@types/react": "18.2.0",
Expand Down
2 changes: 1 addition & 1 deletion packages/app/src/cli/services/build/extension.ts
Original file line number Diff line number Diff line change
Expand Up @@ -118,7 +118,7 @@ export async function buildUIExtension(extension: ExtensionInstance, options: Ex
options.stdout.write(`${extension.localIdentifier} successfully built`)
}

export type BuildFunctionExtensionOptions = ExtensionBuildOptions
type BuildFunctionExtensionOptions = ExtensionBuildOptions

/**
* Builds a function extension
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import {AppEvent, AppEventWatcher, EventType, ExtensionEvent} from './app-event-watcher.js'
import {OutputContextOptions, WatcherEvent, startFileWatcher} from './file-watcher.js'
import {OutputContextOptions, WatcherEvent, FileWatcher} from './file-watcher.js'
import {ESBuildContextManager} from './app-watcher-esbuild.js'
import {
testApp,
Expand All @@ -12,14 +12,14 @@ import {
} from '../../../models/app/app.test-data.js'
import {ExtensionInstance} from '../../../models/extensions/extension-instance.js'
import {loadApp} from '../../../models/app/loader.js'
import {AppInterface} from '../../../models/app/app.js'
import {describe, expect, test, vi} from 'vitest'
import {AbortSignal} from '@shopify/cli-kit/node/abort'
import {flushPromises} from '@shopify/cli-kit/node/promises'
import {inTemporaryDirectory} from '@shopify/cli-kit/node/fs'
import {joinPath} from '@shopify/cli-kit/node/path'
import {Writable} from 'stream'

vi.mock('./file-watcher.js')
vi.mock('../../../models/app/loader.js')
vi.mock('./app-watcher-esbuild.js')

Expand Down Expand Up @@ -234,7 +234,6 @@ describe('app-event-watcher when receiving a file event', () => {
// Given
await inTemporaryDirectory(async (tmpDir) => {
vi.mocked(loadApp).mockResolvedValue(testApp({allExtensions: finalExtensions}))
vi.mocked(startFileWatcher).mockImplementation(async (app, options, onChange) => onChange([fileWatchEvent]))

const buildOutputPath = joinPath(tmpDir, '.shopify', 'bundle')

Expand All @@ -244,7 +243,9 @@ describe('app-event-watcher when receiving a file event', () => {
configuration: {scopes: '', extension_directories: [], path: 'shopify.app.custom.toml'},
})

const watcher = new AppEventWatcher(app, 'url', buildOutputPath, new MockESBuildContextManager())
const mockManager = new MockESBuildContextManager()
const mockFileWatcher = new MockFileWatcher(app, outputOptions, [fileWatchEvent])
const watcher = new AppEventWatcher(app, 'url', buildOutputPath, mockManager, mockFileWatcher)
const emitSpy = vi.spyOn(watcher, 'emit')
await watcher.start()

Expand Down Expand Up @@ -310,7 +311,6 @@ describe('app-event-watcher build extension errors', () => {
extensionPath: '/extensions/ui_extension_1',
startTime: [0, 0],
}
vi.mocked(startFileWatcher).mockImplementation(async (app, options, onChange) => onChange([fileWatchEvent]))

// Given
const esbuildError = {
Expand All @@ -330,9 +330,10 @@ describe('app-event-watcher build extension errors', () => {
allExtensions: [extension1],
configuration: {scopes: '', extension_directories: [], path: 'shopify.app.custom.toml'},
})
const mockFileWatcher = new MockFileWatcher(app, outputOptions, [fileWatchEvent])

// When
const watcher = new AppEventWatcher(app, 'url', buildOutputPath, mockManager)
const watcher = new AppEventWatcher(app, 'url', buildOutputPath, mockManager, mockFileWatcher)
const stderr = {write: vi.fn()} as unknown as Writable
const stdout = {write: vi.fn()} as unknown as Writable

Expand Down Expand Up @@ -363,7 +364,6 @@ describe('app-event-watcher build extension errors', () => {
extensionPath: '/extensions/flow_action',
startTime: [0, 0],
}
vi.mocked(startFileWatcher).mockImplementation(async (app, options, onChange) => onChange([fileWatchEvent]))

// Given
const esbuildError = {message: 'Build failed'}
Expand All @@ -376,7 +376,9 @@ describe('app-event-watcher build extension errors', () => {
})

// When
const watcher = new AppEventWatcher(app, 'url', buildOutputPath, new MockESBuildContextManager())
const mockManager = new MockESBuildContextManager()
const mockFileWatcher = new MockFileWatcher(app, outputOptions, [fileWatchEvent])
const watcher = new AppEventWatcher(app, 'url', buildOutputPath, mockManager, mockFileWatcher)
const stderr = {write: vi.fn()} as unknown as Writable
const stdout = {write: vi.fn()} as unknown as Writable

Expand Down Expand Up @@ -408,3 +410,25 @@ class MockESBuildContextManager extends ESBuildContextManager {
async updateContexts(appEvent: AppEvent) {}
async deleteContexts(extensions: ExtensionInstance[]) {}
}

// Mock class for FileWatcher
// Used to trigger mocked file system events immediately after the watcher is started.
class MockFileWatcher extends FileWatcher {
private readonly events: WatcherEvent[]
private listener?: (events: WatcherEvent[]) => void

constructor(app: AppInterface, options: OutputContextOptions, events: WatcherEvent[]) {
super(app, options)
this.events = events
}

async start(): Promise<void> {
if (this.listener) {
this.listener(this.events)
}
}

onChange(listener: (events: WatcherEvent[]) => void) {
this.listener = listener
}
}
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
/* eslint-disable tsdoc/syntax */
import {OutputContextOptions, startFileWatcher} from './file-watcher.js'
import {FileWatcher, OutputContextOptions} from './file-watcher.js'
import {ESBuildContextManager} from './app-watcher-esbuild.js'
import {handleWatcherEvents} from './app-event-watcher-handler.js'
import {AppLinkedInterface} from '../../../models/app/app.js'
Expand Down Expand Up @@ -96,12 +96,14 @@ export class AppEventWatcher extends EventEmitter {
private started = false
private ready = false
private initialEvents: ExtensionEvent[] = []
private fileWatcher?: FileWatcher

constructor(
app: AppLinkedInterface,
appURL?: string,
buildOutputPath?: string,
esbuildManager?: ESBuildContextManager,
fileWatcher?: FileWatcher,
) {
super()
this.app = app
Expand All @@ -117,6 +119,7 @@ export class AppEventWatcher extends EventEmitter {
url: this.appURL ?? '',
...this.options,
})
this.fileWatcher = fileWatcher
}

async start(options?: OutputContextOptions, buildExtensionsFirst = true) {
Expand All @@ -139,8 +142,8 @@ export class AppEventWatcher extends EventEmitter {
await this.buildExtensions(this.initialEvents)
}

// Start the file system watcher
await startFileWatcher(this.app, this.options, (events) => {
this.fileWatcher = this.fileWatcher ?? new FileWatcher(this.app, this.options)
this.fileWatcher.onChange((events) => {
handleWatcherEvents(events, this.app, this.options)
.then(async (appEvent) => {
if (appEvent?.extensionEvents.length === 0) outputDebug('Change detected, but no extensions were affected')
Expand All @@ -163,6 +166,7 @@ export class AppEventWatcher extends EventEmitter {
this.options.stderr.write(`Error handling event: ${error.message}`)
})
})
await this.fileWatcher.start()

this.ready = true
this.emit('ready', {app: this.app, extensionEvents: this.initialEvents})
Expand Down
17 changes: 13 additions & 4 deletions packages/app/src/cli/services/dev/app-events/file-watcher.test.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import {OutputContextOptions, WatcherEvent, startFileWatcher} from './file-watcher.js'
import {FileWatcher, OutputContextOptions, WatcherEvent} from './file-watcher.js'
import {
testApp,
testAppAccessConfigExtension,
Expand Down Expand Up @@ -198,7 +198,10 @@ describe('file-watcher events', () => {
})

// When
await startFileWatcher(app, outputOptions, vi.fn())
const fileWatcher = new FileWatcher(app, outputOptions)
fileWatcher.onChange(vi.fn())

await fileWatcher.start()

// Then
expect(watchSpy).toHaveBeenCalledWith([joinPath(dir, '/shopify.app.toml'), joinPath(dir, '/extensions')], {
Expand Down Expand Up @@ -232,7 +235,10 @@ describe('file-watcher events', () => {

// When
const onChange = vi.fn()
await startFileWatcher(defaultApp, outputOptions, onChange)
const fileWatcher = new FileWatcher(defaultApp, outputOptions)
fileWatcher.onChange(onChange)

await fileWatcher.start()

// Then
await flushPromises()
Expand Down Expand Up @@ -260,7 +266,10 @@ describe('file-watcher events', () => {

// When
const onChange = vi.fn()
await startFileWatcher(defaultApp, outputOptions, onChange)
const fileWatcher = new FileWatcher(defaultApp, outputOptions)
fileWatcher.onChange(onChange)

await fileWatcher.start()

// Then
await flushPromises()
Expand Down
Loading

0 comments on commit 1b4002b

Please sign in to comment.