Files
cypress/packages/server/lib/socket-e2e.ts
2023-08-25 21:05:33 -05:00

140 lines
4.4 KiB
TypeScript

import Debug from 'debug'
import preprocessor from './plugins/preprocessor'
import { SocketBase } from './socket-base'
import { fs } from './util/fs'
import type { DestroyableHttpServer } from './util/server_destroy'
import * as studio from './studio'
import type { FoundSpec } from '@packages/types'
const debug = Debug('cypress:server:socket-e2e')
const isSpecialSpec = (name) => {
return name.endsWith('__all')
}
export class SocketE2E extends SocketBase {
private testFilePath: string | null
constructor (config: Record<string, any>) {
super(config)
this.testFilePath = null
this.onTestFileChange = this.onTestFileChange.bind(this)
this.onStudioTestFileChange = this.onStudioTestFileChange.bind(this)
this.removeOnStudioTestFileChange = this.removeOnStudioTestFileChange.bind(this)
if (config.watchForFileChanges) {
preprocessor.emitter.on('file:updated', this.onTestFileChange)
}
}
onStudioTestFileChange (filePath) {
// wait for the studio test file to be written to disk, then reload the test
// and remove the listener (since this handler is only invoked when watchForFileChanges is false)
return this.onTestFileChange(filePath).then(() => {
this.removeOnStudioTestFileChange()
})
}
removeOnStudioTestFileChange () {
return preprocessor.emitter.off('file:updated', this.onStudioTestFileChange)
}
onTestFileChange = (filePath) => {
debug('test file changed %o', filePath)
return fs.statAsync(filePath)
.then(() => {
this._cdpIo?.emit('watched:file:changed')
this._socketIo?.emit('watched:file:changed')
}).catch(() => {
return debug('could not find test file that changed %o', filePath)
})
}
watchTestFileByPath (config, specConfig: FoundSpec) {
debug('watching spec with config %o', specConfig)
// previously we have assumed that we pass integration spec path with "integration/" prefix
// now we pass spec config object that tells what kind of spec it is, has relative path already
// so the only special handling remains for special paths like "integration/__all"
// bail if this is special path like "__all"
// maybe the client should not ask to watch non-spec files?
if (isSpecialSpec(specConfig.relative)) {
return
}
if (specConfig.relative.startsWith('/')) {
specConfig.relative = specConfig.relative.slice(1)
}
// bail if we're already watching this exact file
if (specConfig.relative === this.testFilePath) {
return
}
// remove the existing file by its path
if (this.testFilePath) {
preprocessor.removeFile(this.testFilePath, config)
}
// store this location
this.testFilePath = specConfig.relative
debug('will watch test file path %o', specConfig.relative)
return preprocessor.getFile(specConfig.relative, config)
// ignore errors b/c we're just setting up the watching. errors
// are handled by the spec controller
.catch(() => {})
}
startListening (server: DestroyableHttpServer, automation, config, options) {
return super.startListening(server, automation, config, options, {
onSocketConnection: (socket) => {
socket.on('watch:test:file', (specInfo: FoundSpec, cb = function () { }) => {
debug('watch:test:file %o', specInfo)
this.watchTestFileByPath(config, specInfo)
// callback is only for testing purposes
return cb()
})
socket.on('studio:save', (saveInfo, cb) => {
// even if the user has turned off file watching
// we want to force a reload on save
if (!config.watchForFileChanges) {
preprocessor.emitter.on('file:updated', this.onStudioTestFileChange)
}
studio.save(saveInfo)
.then((err) => {
cb(err)
// onStudioTestFileChange will remove itself after being called
// but if there's an error, it never gets called so we manually remove it
if (err && !config.watchForFileChanges) {
this.removeOnStudioTestFileChange()
}
})
.catch(() => {})
})
socket.on('studio:get:commands:text', (commands, cb) => {
const commandsText = studio.getCommandsText(commands)
cb(commandsText)
})
},
})
}
close () {
preprocessor.emitter.removeListener('file:updated', this.onTestFileChange)
return super.close()
}
}