Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: swap websocket transport with cdp add binding/evaluate #27592

Merged
merged 136 commits into from
Aug 26, 2023
Merged
Show file tree
Hide file tree
Changes from 117 commits
Commits
Show all changes
136 commits
Select commit Hold shift + click to select a range
89e7a21
let's do this -- run ci
ryanthemanuel Aug 17, 2023
90cc3a3
try testing part 2
ryanthemanuel Aug 17, 2023
1e222ed
disable auto cancellation
ryanthemanuel Aug 17, 2023
d4b3857
empty commit -- run ci
ryanthemanuel Aug 17, 2023
7125311
let's see how much farther we get
ryanthemanuel Aug 17, 2023
8f980f0
empty commit -- run ci
ryanthemanuel Aug 17, 2023
8128881
empty commit -- run ci
ryanthemanuel Aug 17, 2023
dfdeb6e
empty commit -- run ci
ryanthemanuel Aug 17, 2023
ca9735e
empty commit -- run ci
ryanthemanuel Aug 17, 2023
6271259
split out ios -- run ci
ryanthemanuel Aug 17, 2023
7263ab4
cy in cy -- run ci
ryanthemanuel Aug 18, 2023
5423aee
fix some tests -- run ci
ryanthemanuel Aug 18, 2023
df3f566
fix some tests -- run ci
ryanthemanuel Aug 18, 2023
402dbf4
update interface -- run ci
ryanthemanuel Aug 18, 2023
4bdd17d
update interface -- run ci
ryanthemanuel Aug 18, 2023
dcfa17c
use socket io parser -- run ci
ryanthemanuel Aug 18, 2023
5caae5b
undo refactor -- run ci
ryanthemanuel Aug 18, 2023
fc65eae
fix clean up
ryanthemanuel Aug 18, 2023
7a51835
fix some tests -- run ci
ryanthemanuel Aug 18, 2023
8ddcf59
better cleanup -- run ci
ryanthemanuel Aug 18, 2023
35c9085
fix build
ryanthemanuel Aug 18, 2023
5446a05
fix some tests -- run ci
ryanthemanuel Aug 18, 2023
df8f99e
skip tests -- run ci
ryanthemanuel Aug 18, 2023
33a47c8
skip tests -- run ci
ryanthemanuel Aug 18, 2023
faed5ef
skip tests -- run ci
ryanthemanuel Aug 18, 2023
c65f598
skip tests -- run ci
ryanthemanuel Aug 18, 2023
e62fcc1
skip tests -- run ci
ryanthemanuel Aug 18, 2023
fc08795
skip tests -- run ci
ryanthemanuel Aug 18, 2023
57a623c
fix tests -- run ci
ryanthemanuel Aug 18, 2023
8e78f67
skip tests -- run ci
ryanthemanuel Aug 19, 2023
7f313fa
skip tests -- run ci
ryanthemanuel Aug 19, 2023
a1fb622
try to recreate with debugging
ryanthemanuel Aug 19, 2023
5648fc0
try and await -- run ci
ryanthemanuel Aug 19, 2023
7e9f9dd
try and await -- run ci
ryanthemanuel Aug 19, 2023
40f035e
try and await -- run ci
ryanthemanuel Aug 19, 2023
2a75906
Update cri-client.ts
ryanthemanuel Aug 19, 2023
19a7147
try and await -- run ci
ryanthemanuel Aug 19, 2023
e7afa7e
try and await -- run ci
ryanthemanuel Aug 19, 2023
11f971a
better impl
ryanthemanuel Aug 19, 2023
64c2467
better impl
ryanthemanuel Aug 19, 2023
51e2b29
back out experiment
ryanthemanuel Aug 19, 2023
09ad9c4
binary data
ryanthemanuel Aug 20, 2023
6bc5bd3
unskip tests
ryanthemanuel Aug 20, 2023
a97ba99
fix crash on reconnect
ryanthemanuel Aug 20, 2023
0bc2887
fix integration test
ryanthemanuel Aug 20, 2023
db21262
add debugging
ryanthemanuel Aug 20, 2023
f3719d5
fix user-agent not being available when using the CDPSocket
brian-mann Aug 20, 2023
8ccc6d6
Merge branch 'ryanm/feat/swap-websockets-with-cdp' of https://github.…
brian-mann Aug 20, 2023
9e8d4f3
move setBrowserStatus to a data-context action since it's a mutation
brian-mann Aug 20, 2023
d332e0b
sort import list
brian-mann Aug 20, 2023
020cb21
remove lots of indirection that was hoisting arbitrary getters to dat…
brian-mann Aug 20, 2023
bf5f093
properly update projects with ctx.update, remove unused private getter
brian-mann Aug 20, 2023
9601df0
refactor server actions into their own data context class
brian-mann Aug 20, 2023
21d87f9
fix integration tests with the userAgent change
brian-mann Aug 20, 2023
3794a42
fix property accessors
brian-mann Aug 20, 2023
700e072
update setters to use actions
brian-mann Aug 20, 2023
cdfedbd
cleanup function refs
brian-mann Aug 20, 2023
dc4c44f
invoke the promisified destroy function
brian-mann Aug 20, 2023
5d99044
update debug statements to log out the target
brian-mann Aug 21, 2023
7a27d3f
improve cleanup
ryanthemanuel Aug 21, 2023
390a9c3
fix test
ryanthemanuel Aug 21, 2023
5b99df0
fix tests
ryanthemanuel Aug 21, 2023
c65de0f
fix next tick issue
ryanthemanuel Aug 21, 2023
c6f3bf7
add debugging
ryanthemanuel Aug 21, 2023
b79cce7
try and fix clean up logic
ryanthemanuel Aug 21, 2023
d0403e4
fix electron
ryanthemanuel Aug 22, 2023
64288e7
Merge branch 'release/13.0.0' into ryanm/feat/swap-websockets-with-cdp
ryanthemanuel Aug 22, 2023
47edc94
add debuggin
ryanthemanuel Aug 22, 2023
881c1bb
debugging
ryanthemanuel Aug 22, 2023
5e29a84
use single line options debug
brian-mann Aug 22, 2023
27b62f6
add missing system test for when the tab/page is closed
brian-mann Aug 22, 2023
1f6d635
ensure verbose debug logs when cdp msgs are received are logged first…
brian-mann Aug 22, 2023
0f944b6
prevent race conditional calling close twice hangs the close promise
brian-mann Aug 22, 2023
7c2eb5e
add tests for handling unexpected Browser.close events
brian-mann Aug 22, 2023
de84b07
refactor connecting to CDP socket, encapsulate local variables
brian-mann Aug 22, 2023
b1ce42c
add new BrowserCriClient state to manage unexpected crashes for the b…
brian-mann Aug 22, 2023
0e95b31
temporarily comment out reconnecting on a disconnect
brian-mann Aug 22, 2023
48efe23
begin implementation handling recovery from unexpected browser or pag…
brian-mann Aug 23, 2023
87e2cde
exit the run completely when the browser process closes
brian-mann Aug 23, 2023
3163dc9
add error snapshots
brian-mann Aug 23, 2023
c53f351
fix origin tests
ryanthemanuel Aug 23, 2023
fbf0782
fix bluebird
ryanthemanuel Aug 23, 2023
205d7cf
add test for page closed - chrome
ryanthemanuel Aug 23, 2023
d311f9d
fix firefox
ryanthemanuel Aug 23, 2023
d8831de
slight refactor
ryanthemanuel Aug 23, 2023
18c2de3
rework logic to create signals for whether a graceful shutdown is hap…
brian-mann Aug 23, 2023
69f3dbd
do not check for browser/page closes when targetDestroyed is for a pr…
brian-mann Aug 23, 2023
5a29dfb
retries
ryanthemanuel Aug 23, 2023
ad3b8fa
code cleanup
ryanthemanuel Aug 24, 2023
cb5f7ba
minor rework
ryanthemanuel Aug 24, 2023
ec6d1a1
fix unit tests
ryanthemanuel Aug 24, 2023
2661609
Update browser-cri-client.ts
ryanthemanuel Aug 24, 2023
4e0e971
fix issue with hanging
ryanthemanuel Aug 24, 2023
5580360
minor cleanup
ryanthemanuel Aug 24, 2023
e211198
PR comments
ryanthemanuel Aug 24, 2023
90e21cd
Update electron.ts
ryanthemanuel Aug 25, 2023
277926b
Update electron.ts
ryanthemanuel Aug 25, 2023
981def5
fix retry issue
ryanthemanuel Aug 25, 2023
a22ac87
fix retry issue
ryanthemanuel Aug 25, 2023
9ce649e
revert
ryanthemanuel Aug 25, 2023
c84364f
fix reconnect issue
ryanthemanuel Aug 25, 2023
5597dcb
fix retry
ryanthemanuel Aug 25, 2023
7b4d082
remove test for electron browser crashing bc its invalid
brian-mann Aug 25, 2023
62ad688
Update electron.ts
ryanthemanuel Aug 25, 2023
3c7b959
fix cy in cy problems
ryanthemanuel Aug 25, 2023
ae40d09
fix cy in cy problems
ryanthemanuel Aug 25, 2023
0b1b76c
Update packages/server/lib/browsers/electron.ts
ryanthemanuel Aug 25, 2023
4e1085d
Merge remote-tracking branch 'origin/ryanm/feat/swap-websockets-with-…
ryanthemanuel Aug 25, 2023
13e8218
fix unit tests
ryanthemanuel Aug 25, 2023
49571e7
update workflows.yml
ryanthemanuel Aug 25, 2023
faf7cfe
Apply suggestions from code review
ryanthemanuel Aug 25, 2023
ec0038b
Merge branch 'release/13.0.0' into ryanm/feat/swap-websockets-with-cdp
ryanthemanuel Aug 25, 2023
fbfbbbf
experiment
ryanthemanuel Aug 25, 2023
ede92c7
merge release/13.0.0
ryanthemanuel Aug 25, 2023
9e525da
add integration tests for cdp socket reconnection
brian-mann Aug 25, 2023
26c7382
Merge branch 'ryanm/feat/swap-websockets-with-cdp' of https://github.…
brian-mann Aug 25, 2023
f8ffb23
Update packages/socket/lib/cdp-socket.ts
ryanthemanuel Aug 25, 2023
e60c3bd
add tests for stopping the reconnection after close is called
brian-mann Aug 25, 2023
75cdd0d
Merge branch 'ryanm/feat/swap-websockets-with-cdp' of https://github.…
brian-mann Aug 25, 2023
bbedffa
add error handling
ryanthemanuel Aug 25, 2023
3aa702a
when the spec closes in run mode always abort the run
brian-mann Aug 25, 2023
995cd60
Merge branch 'ryanm/feat/swap-websockets-with-cdp' of https://github.…
brian-mann Aug 25, 2023
3138c2b
support overriding system test timeout with env var
brian-mann Aug 25, 2023
b4c9b4c
update with debug logs on errors
ryanthemanuel Aug 25, 2023
c04a7ce
update error message on browser/tab crash to clarify which browserNam…
brian-mann Aug 25, 2023
a82fc69
remove unnecessary handler on Target.targetCrashed
brian-mann Aug 25, 2023
9e707c8
Merge branch 'ryanm/feat/swap-websockets-with-cdp' of https://github.…
brian-mann Aug 25, 2023
842ff37
remove test for electron crashing bc it hangs the entire cypress proc…
brian-mann Aug 25, 2023
aa9d1d6
let unhandled unrecoverable errors bubble up
ryanthemanuel Aug 25, 2023
35084dd
add note about why we aren't testing browser crashes in electron [ski…
brian-mann Aug 25, 2023
72a908b
Merge branch 'ryanm/feat/swap-websockets-with-cdp' of https://github.…
brian-mann Aug 25, 2023
3159a09
fix errors not exiting on uncaught exception, fixed error
brian-mann Aug 25, 2023
9f5a610
fix typescript
ryanthemanuel Aug 25, 2023
56a8744
Merge branch 'release/13.0.0' into ryanm/feat/swap-websockets-with-cdp
ryanthemanuel Aug 25, 2023
4ef0f44
fix builds
ryanthemanuel Aug 26, 2023
ede32c5
fix test
ryanthemanuel Aug 26, 2023
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
57 changes: 0 additions & 57 deletions packages/app/cypress/e2e/cypress-in-cypress-component.cy.ts
Original file line number Diff line number Diff line change
Expand Up @@ -178,63 +178,6 @@ describe('Cypress In Cypress CT', { viewportWidth: 1500, defaultCommandTimeout:
expect(ctx.actions.project.initializeActiveProject).to.be.called
})
})

it('moves away from runner and back, disconnects websocket and reconnects it correctly', () => {
cy.openProject('cypress-in-cypress')
cy.startAppServer('component')

cy.visitApp()
cy.contains('TestComponent.spec').click()
cy.waitForSpecToFinish()
cy.get('[data-model-state="passed"]').should('contain', 'renders the test component')
cy.get('.passed > .num').should('contain', 1)
cy.get('.failed > .num').should('contain', '--')

cy.findByTestId('sidebar-link-runs-page').click()
cy.get('[data-cy="app-header-bar"]').findByText('Runs').should('be.visible')

cy.findByTestId('sidebar-link-specs-page').click()
cy.get('[data-cy="app-header-bar"]').findByText('Specs').should('be.visible')

cy.contains('TestComponent.spec').click()
cy.waitForSpecToFinish()
cy.get('[data-model-state="passed"]').should('contain', 'renders the test component')

cy.window().then((win) => {
const connected = () => win.ws?.connected

win.ws?.close()

cy.wrap({
connected,
}).invoke('connected').should('be.false')

win.ws?.connect()

cy.wrap({
connected,
}).invoke('connected').should('be.true')
})

cy.withCtx(async (ctx, o) => {
await ctx.actions.file.writeFileInProject(o.path, `
import React from 'react'
import { mount } from 'cypress/react'

describe('TestComponent', () => {
it('renders the new test component', () => {
mount(<div>Component Test</div>)

cy.contains('Component Test').should('be.visible')
})
})
`)
}, { path: getPathForPlatform('src/TestComponent.spec.jsx') })

cy.get('[data-model-state="passed"]').should('contain', 'renders the new test component')
cy.get('.passed > .num').should('contain', 1)
cy.get('.failed > .num').should('contain', '--')
})
})

context('custom config', () => {
Expand Down
49 changes: 0 additions & 49 deletions packages/app/cypress/e2e/cypress-in-cypress-e2e.cy.ts
Original file line number Diff line number Diff line change
Expand Up @@ -231,55 +231,6 @@ describe('Cypress In Cypress E2E', { viewportWidth: 1500, defaultCommandTimeout:
cy.get('[data-model-state="passed"]').should('contain', 'expected true to be true')
})

it('moves away from runner and back, disconnects websocket and reconnects it correctly', () => {
cy.visitApp()
cy.contains('dom-content.spec').click()
cy.waitForSpecToFinish()
cy.get('[data-model-state="passed"]').should('contain', 'renders the test content')
cy.get('.passed > .num').should('contain', 1)
cy.get('.failed > .num').should('contain', '--')

cy.findByTestId('sidebar-link-runs-page').click()
cy.get('[data-cy="app-header-bar"]').findByText('Runs').should('be.visible')

cy.findByTestId('sidebar-link-specs-page').click()
cy.get('[data-cy="app-header-bar"]').findByText('Specs').should('be.visible')

cy.contains('dom-content.spec').click()
cy.waitForSpecToFinish()
cy.get('[data-model-state="passed"]').should('contain', 'renders the test content')

cy.window().then((win) => {
const connected = () => win.ws?.connected

win.ws?.close()

cy.wrap({
connected,
}).invoke('connected').should('be.false')

win.ws?.connect()

cy.wrap({
connected,
}).invoke('connected').should('be.true')
})

cy.withCtx(async (ctx, o) => {
await ctx.actions.file.writeFileInProject(o.path, `
describe('Dom Content', () => {
it('renders the new test content', () => {
cy.visit('cypress/e2e/dom-content.html')
})
})
`)
}, { path: getPathForPlatform('cypress/e2e/dom-content.spec.js') })

cy.get('[data-model-state="passed"]').should('contain', 'renders the new test content')
cy.get('.passed > .num').should('contain', 1)
cy.get('.failed > .num').should('contain', '--')
})

describe('accessibility', () => {
it('has no axe violations in specs list panel', () => {
cy.visitApp()
Expand Down
4 changes: 2 additions & 2 deletions packages/app/cypress/e2e/cypress-in-cypress.cy.ts
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@ describe('Cypress in Cypress', { viewportWidth: 1500, defaultCommandTimeout: 100
cy.waitForSpecToFinish()

cy.withCtx((ctx) => {
ctx.coreData.servers.appSocketServer?.emit('automation:disconnected')
ctx.coreData.servers.cdpSocketServer?.emit('automation:disconnected')
})

cy.contains('h3', 'The Cypress extension has disconnected')
Expand Down Expand Up @@ -403,7 +403,7 @@ describe('Cypress in Cypress', { viewportWidth: 1500, defaultCommandTimeout: 100
cy.withCtx(async (ctx) => {
const currentProject = ctx.currentProject?.replaceAll('\\', '/')
const specPath = `${currentProject}/cypress/e2e/dom-content.spec.js`
const url = `http://127.0.0.1:${ctx.gqlServerPort}/__launchpad/graphql?`
const url = `http://127.0.0.1:${ctx.coreData.servers.gqlServerPort}/__launchpad/graphql?`
const payload = `{"query":"mutation{\\nrunSpec(specPath:\\"${specPath}\\"){\\n__typename\\n... on RunSpecResponse{\\ntestingType\\nbrowser{\\nid\\nname\\n}\\nspec{\\nid\\nname\\n}\\n}\\n}\\n}","variables":null}`

ctx.coreData.app.browserStatus = 'open'
Expand Down
9 changes: 5 additions & 4 deletions packages/app/cypress/e2e/runs.cy.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import defaultMessages from '@packages/frontend-shared/src/locales/en-US.json'

import type { SinonStub } from 'sinon'

function moveToRunsPage (): void {
Expand Down Expand Up @@ -616,26 +617,26 @@ describe('App: Runs', { viewportWidth: 1200 }, () => {
it('displays a copy button and copies correct command in Component Testing', () => {
scaffoldTestingTypeAndVisitRunsPage('component')
cy.withCtx(async (ctx, o) => {
o.sinon.stub(ctx.electronApi, 'copyTextToClipboard')
o.sinon.stub(ctx.config.electronApi, 'copyTextToClipboard')
})

cy.get('[data-cy="copy-button"]').click()
cy.contains('Copied!')
cy.withRetryableCtx((ctx) => {
expect(ctx.electronApi.copyTextToClipboard as SinonStub).to.have.been.calledWith('npx cypress run --component --record --key 2aaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa')
expect(ctx.config.electronApi.copyTextToClipboard as SinonStub).to.have.been.calledWith('npx cypress run --component --record --key 2aaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa')
})
})

it('displays a copy button and copies correct command in E2E', () => {
scaffoldTestingTypeAndVisitRunsPage('e2e')
cy.withCtx(async (ctx, o) => {
o.sinon.stub(ctx.electronApi, 'copyTextToClipboard')
o.sinon.stub(ctx.config.electronApi, 'copyTextToClipboard')
})

cy.get('[data-cy="copy-button"]').click()
cy.contains('Copied!')
cy.withRetryableCtx((ctx) => {
expect(ctx.electronApi.copyTextToClipboard as SinonStub).to.have.been.calledWith('npx cypress run --record --key 2aaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa')
expect(ctx.config.electronApi.copyTextToClipboard as SinonStub).to.have.been.calledWith('npx cypress run --record --key 2aaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa')
})
})
})
Expand Down
4 changes: 2 additions & 2 deletions packages/app/cypress/e2e/settings.cy.ts
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ describe('App: Settings', () => {
describe('Cloud Settings', () => {
it('shows the projectId section when there is a projectId and shows override from CLI', () => {
cy.withCtx(async (ctx, o) => {
o.sinon.stub(ctx.electronApi, 'copyTextToClipboard')
o.sinon.stub(ctx.config.electronApi, 'copyTextToClipboard')
})

cy.startAppServer('e2e')
Expand All @@ -40,7 +40,7 @@ describe('App: Settings', () => {
cy.findByText('Copy').click()
cy.findByText('Copied!').should('be.visible')
cy.withRetryableCtx((ctx) => {
expect(ctx.electronApi.copyTextToClipboard as SinonStub).to.have.been.calledWith('fromCli')
expect(ctx.config.electronApi.copyTextToClipboard as SinonStub).to.have.been.calledWith('fromCli')
})
})

Expand Down
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import defaultMessages from '@packages/frontend-shared/src/locales/en-US.json'

import type { SinonStub } from 'sinon'

describe('CreateCloudOrgModalSubscription', { viewportWidth: 1200 }, () => {
Expand Down Expand Up @@ -51,7 +52,7 @@ describe('CreateCloudOrgModalSubscription', { viewportWidth: 1200 }, () => {
})

cy.withCtx(async (ctx) => {
await ctx.util.fetch(`http://127.0.0.1:${ctx.gqlServerPort}/cloud-notification?operationName=orgCreated`)
await ctx.util.fetch(`http://127.0.0.1:${ctx.coreData.servers.gqlServerPort}/cloud-notification?operationName=orgCreated`)
})

cy.findByText(defaultMessages.runs.connect.modal.selectProject.manageOrgs)
Expand Down
4 changes: 2 additions & 2 deletions packages/app/index.d.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
/// <reference path="../driver/types/internal-types-lite.d.ts" />

import type { Socket } from '@packages/socket/lib/browser'
import type { SocketShape } from '@packages/socket/lib/types'
import type MobX from 'mobx'
import type { EventManager } from './src/runner/event-manager'

Expand All @@ -21,7 +21,7 @@ export {}
*/
declare global {
interface Window {
ws?: Socket
ws?: SocketShape
getEventManager: () => EventManager
UnifiedRunner: {
/**
Expand Down
8 changes: 4 additions & 4 deletions packages/app/src/runner/event-manager.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ import type { LocalBusEmitsMap, LocalBusEventMap, DriverToLocalBus, SocketToDriv
import type { RunState, CachedTestState, AutomationElementId, FileDetails, ReporterStartInfo, ReporterRunState } from '@packages/types'

import { logger } from './logger'
import type { Socket } from '@packages/socket/lib/browser'
import type { SocketShape } from '@packages/socket/lib/types'
import { automation, useRunnerUiStore, useSpecStore } from '../store'
import { useScreenshotStore } from '../store/screenshot-store'
import { useStudioStore } from '../store/studio-store'
Expand Down Expand Up @@ -57,7 +57,7 @@ export class EventManager {
selectorPlaygroundModel: any
cypressInCypressMochaEvents: CypressInCypressMochaEvent[] = []
// Used for testing the experimentalSingleTabRunMode experiment. Ensures AUT is correctly destroyed between specs.
ws: Socket
ws: SocketShape
specStore: ReturnType<typeof useSpecStore>
studioStore: ReturnType<typeof useStudioStore>

Expand All @@ -68,7 +68,7 @@ export class EventManager {
private Mobx: typeof MobX,
// selectorPlaygroundModel singleton
selectorPlaygroundModel: any,
ws: Socket,
ws: SocketShape,
) {
this.selectorPlaygroundModel = selectorPlaygroundModel
this.ws = ws
Expand Down Expand Up @@ -134,7 +134,7 @@ export class EventManager {
telemetry.setRootContext(context)
})

this.ws.on('automation:push:message', (msg, data = {}) => {
this.ws.on('automation:push:message', (msg, data: any = {}) => {
if (!Cypress) return

switch (msg) {
Expand Down
6 changes: 6 additions & 0 deletions packages/data-context/src/DataActions.ts
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ import {
BrowserActions,
DevActions,
AuthActions,
ServersActions,
CohortsActions,
CodegenActions,
CloudProjectActions,
Expand Down Expand Up @@ -78,6 +79,11 @@ export class DataActions {
return new BrowserActions(this.ctx)
}

@cached
get servers () {
return new ServersActions(this.ctx)
}

@cached
get versions () {
return new VersionsActions(this.ctx)
Expand Down
Loading