* basic card visuals, working on click * visual tweaks and i18n * match other i18n * redo duration format, should make common too * basic debug, needs validation of relevant run * refactor and move files * working top level state * working grouping, needs subcomponent * switch click target * remove dayjs change * fix current commit * revise commit for message * working view runs button * fix type check * fix tests * fix integration tests * remove old test * add layout * add component test for new layout and cleanup * add count check * standardize cy tags * revise count test * fix typos * fix padding * start wrapping * happy wrap, next trunc * responsive if wonky * clean up and titles * magic resize, needs tests per breakpoint * standardize the rollup * working default rollup * rollup spacing tweaks * center elements * update tests for multiple viewports * add missing breakpoint tests * fix debug tests for status and badge * remove unneeded check * add functional component * fix check_ts * update container test * add changelog * add tooltip, needs aria label and external link update * fix overflow * fix avatar and add aria label to count summary * change to external href * remove popup sizing * add tests * add missing data cy * fix runcard test * Truncate result count * make truncate style conditional * match design truncate position * adddress PR comments * chore:updating styles (#27798) * add tooltip content tests and skeletons * clean up tooltip validation * fix runs selector * Update packages/app/src/runs/RunsSkeletonRow.vue Co-authored-by: Stokes Player <stokes@cypress.io> * Update packages/app/src/runs/RunTagCount.vue Co-authored-by: Stokes Player <stokes@cypress.io> * Update packages/app/src/runs/RunsSkeleton.vue Co-authored-by: Stokes Player <stokes@cypress.io> * Update packages/app/src/runs/RunTagCount.vue Co-authored-by: Stokes Player <stokes@cypress.io> * Update packages/app/src/runs/RunsSkeleton.vue Co-authored-by: Stokes Player <stokes@cypress.io> * Update packages/app/src/runs/RunsSkeletonRow.vue Co-authored-by: Stokes Player <stokes@cypress.io> * Update packages/app/src/runs/RunTagCount.vue Co-authored-by: Stokes Player <stokes@cypress.io> * Update packages/app/src/runs/RunTagCount.vue Co-authored-by: Stokes Player <stokes@cypress.io> * remove redundant shrink-1s * more shrinks --------- Co-authored-by: Stokes Player <stokes@cypress.io>
@packages/data-context
Centralized data access for the Cypress application
Directories & Concepts
There are several directories in src:
actionscodegendatagensourcesutil
The main ones you need to know about are data, sources and actions.
Here are some general guidelines associated with each, and an example showing how they are used together.
Data
This contains the interfaces that describe the top level data (called coreData) that is exposed and used by launchpad and app. Secondary data that isn't exposed to the outside world (temporary states, flags, etc) is usually in a Source. Sources are also used to derive data.
If you want to update Data, you use an Action (see below).
Sources
The sources directory contains what can be thought of as "read only" and "derived" data. Each one is namespaced based on the kind of data it's associated with, for example Project, Browser, Settings, etc. Sources can access the ctx (type DataContext, see DataContext.ts), using this.ctx.
If you want to update something in a Source, or in coreData, you want to do it using an Action.
Actions
Actions are where mutative and destructive operations live. To make this predictable and changes each to track, updating this.ctx.coreData should be done via an Action and use this.ctx.update, which receives the current coreData as the first argument.
Example
In this example, we will load some specs for a project and persist them. We will use a Source to derive any specs with the characters "foo" in the filename. This shows how Data, Sources and Actions are connected.
1. Define Data data/coreData
First we define the type in CoreDataShape and set the initial value in makeCoreData.
export interface CoreDataShape {
specs: string[]
}
// ...
export function makeCoreData (modeOptions: Partial<AllModeOptions> = {}): CoreDataShape {
return {
// ...
specs: [],
}
}
This is where the actual value will be saved.
2. Define Action to Update Specs
We need some way to update the value. For this, we are defining a new SpecActions class inside of actions and updating the coreData with this.ctx.update.
import type { DataContext } from '..'
import globby from 'globby'
export class SpecActions {
constructor (private ctx: DataContext) {}
async findSpecs () {
const specs = await globby('./**/*.spec.js')
this.ctx.update(coreData => {
coreData.specs = specs
})
}
}
Note: If you added a new Action file, you will also need to add it to DataActions.ts, although this isn't very common.
import type { DataContext } from '.'
import {
// ...
SpecActions
} from './actions'
import { cached } from './util'
export class DataActions {
constructor (private ctx: DataContext) {}
// ...
@cached
get specs () {
return new SpecActions(this.ctx)
}
}
3. Derive the Data with a Source
In this example we only want to expose specs with foo in the name. We can derive this using a Source. This will be a new Source call SpecDataSource, but you can use an existing one if it makes sense.
import type { DataContext } from '..'
export class SpecDataSource {
constructor (private ctx: DataContext) {}
fooSpecs () {
return this.ctx.coreData.specs.find(spec => spec.includes('foo'))
}
}
If you added a new Source, you need to add it to DataContext.ts.
import { SpecDataSource } from './sources/SpecDataSource'
export class DataContext {
// ...
@cached
get specs () {
return new SpecDataSource(this)
}
}
4. (Bonus) Expose via GraphQL
You might want to expose your new Data or Source via GraphQL. It's easy, since GraphQL also has access ctx as the third argument to the resolvers. For example, we can expose specs and fooSpecs in gql-Query.ts like this:
export const Query = objectType({
definition (t) {
// ...
t.list.string('specs', {
description: 'A list of specs',
resolve: (source, args, ctx) => {
return ctx.coreData.specs
},
})
t.list.string('fooSpecs', {
description: 'A list of specs containing foo',
resolve: (source, args, ctx) => {
return ctx.specs.fooSpecs()
},
})
}
})