diff --git a/packages/manager/.changeset/pr-11428-tech-stories-1734382681885.md b/packages/manager/.changeset/pr-11428-tech-stories-1734382681885.md new file mode 100644 index 00000000000..319aef8f779 --- /dev/null +++ b/packages/manager/.changeset/pr-11428-tech-stories-1734382681885.md @@ -0,0 +1,5 @@ +--- +"@linode/manager": Tech Stories +--- + +Add MSW crud domains ([#11428](https://github.com/linode/manager/pull/11428)) diff --git a/packages/manager/src/dev-tools/load.ts b/packages/manager/src/dev-tools/load.ts index f77ea890b71..563a50e194d 100644 --- a/packages/manager/src/dev-tools/load.ts +++ b/packages/manager/src/dev-tools/load.ts @@ -87,6 +87,7 @@ export async function loadDevTools( // Merge the contexts const mergedContext: MockState = { ...initialContext, + domains: [...initialContext.domains, ...(seedContext?.domains || [])], eventQueue: [ ...initialContext.eventQueue, ...(seedContext?.eventQueue || []), diff --git a/packages/manager/src/features/Events/FormattedEventMessage.tsx b/packages/manager/src/features/Events/FormattedEventMessage.tsx index f378c933be1..21c6c3d2d79 100644 --- a/packages/manager/src/features/Events/FormattedEventMessage.tsx +++ b/packages/manager/src/features/Events/FormattedEventMessage.tsx @@ -4,6 +4,7 @@ import * as React from 'react'; import { SupportLink } from 'src/components/SupportLink'; interface MessageLinkEntity { + fallback?: string; message: null | string; } @@ -15,10 +16,10 @@ interface MessageLinkEntity { * - render "contact support" strings as . */ export const FormattedEventMessage = (props: MessageLinkEntity) => { - const { message } = props; + const { fallback, message } = props; if (!message) { - return null; + return fallback ? fallback : null; } return formatMessage(message); @@ -34,9 +35,14 @@ const formatMessage = (message: string): JSX.Element => { let formattedPart: JSX.Element | string = part; if (part.startsWith('`') && part.endsWith('`')) { - formattedPart = ( - {part.slice(1, -1)} - ); + const content = part.slice(1, -1); + if (content.length > 0) { + formattedPart = ( + {content} + ); + } else { + formattedPart = ''; + } } if (part.match(supportLinkMatch)) { diff --git a/packages/manager/src/features/Events/factories/domain.tsx b/packages/manager/src/features/Events/factories/domain.tsx index c29d2e95c27..61377e97427 100644 --- a/packages/manager/src/features/Events/factories/domain.tsx +++ b/packages/manager/src/features/Events/factories/domain.tsx @@ -30,12 +30,14 @@ export const domain: PartialEventMap<'domain'> = { ), }, domain_record_create: { - notification: (e) => ( - <> - has been{' '} - added to . - - ), + notification: (e) => { + return ( + <> + has + been added to . + + ); + }, }, domain_record_delete: { notification: (e) => ( diff --git a/packages/manager/src/mocks/mockState.ts b/packages/manager/src/mocks/mockState.ts index d4aa0d3a905..5d95b9f3f46 100644 --- a/packages/manager/src/mocks/mockState.ts +++ b/packages/manager/src/mocks/mockState.ts @@ -22,6 +22,8 @@ export const getStateSeederGroups = ( }; export const emptyStore: MockState = { + domainRecords: [], + domains: [], eventQueue: [], firewalls: [], linodeConfigs: [], diff --git a/packages/manager/src/mocks/presets/baseline/crud.ts b/packages/manager/src/mocks/presets/baseline/crud.ts index 1719f7772fd..12679014c81 100644 --- a/packages/manager/src/mocks/presets/baseline/crud.ts +++ b/packages/manager/src/mocks/presets/baseline/crud.ts @@ -4,6 +4,7 @@ import { } from 'src/mocks/presets/crud/handlers/events'; import { linodeCrudPreset } from 'src/mocks/presets/crud/linodes'; +import { domainCrudPreset } from '../crud/domains'; import { placementGroupsCrudPreset } from '../crud/placementGroups'; import { supportTicketCrudPreset } from '../crud/supportTickets'; import { volumeCrudPreset } from '../crud/volumes'; @@ -17,6 +18,7 @@ export const baselineCrudPreset: MockPresetBaseline = { ...placementGroupsCrudPreset.handlers, ...supportTicketCrudPreset.handlers, ...volumeCrudPreset.handlers, + ...domainCrudPreset.handlers, // Events. getEvents, diff --git a/packages/manager/src/mocks/presets/crud/domains.ts b/packages/manager/src/mocks/presets/crud/domains.ts new file mode 100644 index 00000000000..58a2d136617 --- /dev/null +++ b/packages/manager/src/mocks/presets/crud/domains.ts @@ -0,0 +1,24 @@ +import { + cloneDomain, + createDomain, + deleteDomains, + getDomains, + importDomain, + updateDomain, +} from 'src/mocks/presets/crud/handlers/domains'; + +import type { MockPresetCrud } from 'src/mocks/types'; + +export const domainCrudPreset: MockPresetCrud = { + group: { id: 'Domains' }, + handlers: [ + createDomain, + deleteDomains, + updateDomain, + getDomains, + cloneDomain, + importDomain, + ], + id: 'domains:crud', + label: 'Domains CRUD', +}; diff --git a/packages/manager/src/mocks/presets/crud/handlers/domains.ts b/packages/manager/src/mocks/presets/crud/handlers/domains.ts new file mode 100644 index 00000000000..4c14bd4152b --- /dev/null +++ b/packages/manager/src/mocks/presets/crud/handlers/domains.ts @@ -0,0 +1,252 @@ +import { DateTime } from 'luxon'; +import { http } from 'msw'; + +import { domainFactory, domainRecordFactory } from 'src/factories'; +import { mswDB } from 'src/mocks/indexedDB'; +import { queueEvents } from 'src/mocks/utilities/events'; +import { + makeNotFoundResponse, + makePaginatedResponse, + makeResponse, +} from 'src/mocks/utilities/response'; + +import type { Domain, DomainRecord } from '@linode/api-v4'; +import type { StrictResponse } from 'msw'; +import type { MockState } from 'src/mocks/types'; +import type { + APIErrorResponse, + APIPaginatedResponse, +} from 'src/mocks/utilities/response'; + +export const getDomains = () => [ + http.get( + '*/v4/domains/:id/records', + async ({ + request, + }): Promise< + StrictResponse> + > => { + const domainRecords = domainRecordFactory.buildList(3); + + return makePaginatedResponse({ + data: domainRecords, + request, + }); + } + ), + + http.get( + '*/v4/domains', + async ({ + request, + }): Promise< + StrictResponse> + > => { + const domains = await mswDB.getAll('domains'); + + if (!domains) { + return makeNotFoundResponse(); + } + + return makePaginatedResponse({ + data: domains, + request, + }); + } + ), + + http.get( + '*/v4/domains/:id', + async ({ params }): Promise> => { + const id = Number(params.id); + const domain = await mswDB.get('domains', id); + + if (!domain) { + return makeNotFoundResponse(); + } + + return makeResponse(domain); + } + ), +]; + +export const createDomain = (mockState: MockState) => [ + http.post( + '*/v4/domains', + async ({ request }): Promise> => { + const payload = await request.clone().json(); + + const domain = domainFactory.build({ + ...payload, + created: DateTime.now().toISO(), + last_updated: DateTime.now().toISO(), + updated: DateTime.now().toISO(), + }); + + await mswDB.add('domains', domain, mockState); + + queueEvents({ + event: { + action: 'domain_create', + entity: { + id: domain.id, + label: domain.domain, + type: 'domain', + url: `/v4/domains/${domain.id}`, + }, + }, + mockState, + sequence: [{ status: 'notification' }], + }); + + return makeResponse(domain); + } + ), +]; + +export const updateDomain = (mockState: MockState) => [ + http.put( + '*/v4/domains/:id', + async ({ + params, + request, + }): Promise> => { + const id = Number(params.id); + const domain = await mswDB.get('domains', id); + + if (!domain) { + return makeNotFoundResponse(); + } + + const payload = { + ...(await request.clone().json()), + last_updated: DateTime.now().toISO(), + updated: DateTime.now().toISO(), + }; + const updatedDomain = { ...domain, ...payload }; + + await mswDB.update('domains', id, updatedDomain, mockState); + + queueEvents({ + event: { + action: 'domain_update', + entity: { + id: domain.id, + label: domain.domain, + type: 'domain', + url: `/v4/domains/${domain.id}`, + }, + }, + mockState, + sequence: [{ status: 'notification' }], + }); + + return makeResponse(updatedDomain); + } + ), +]; + +export const cloneDomain = (mockState: MockState) => [ + http.post( + '*/v4/domains/:id/clone', + async ({ params }): Promise> => { + const id = Number(params.id); + const domain = await mswDB.get('domains', id); + + if (!domain) { + return makeNotFoundResponse(); + } + + const clonedDomain = { + ...domain, + created: DateTime.now().toISO(), + last_updated: DateTime.now().toISO(), + updated: DateTime.now().toISO(), + }; + + await mswDB.add('domains', clonedDomain, mockState); + + queueEvents({ + event: { + action: 'domain_create', + entity: { + id: domain.id, + label: domain.domain, + type: 'domain', + url: `/v4/domains/${domain.id}`, + }, + }, + mockState, + sequence: [{ status: 'notification' }], + }); + + return makeResponse(domain); + } + ), +]; + +export const importDomain = (mockState: MockState) => [ + http.post( + '*/v4/domains/import', + async ({ request }): Promise> => { + const payload = await request.clone().json(); + + const domain = domainFactory.build({ + ...payload, + created: DateTime.now().toISO(), + last_updated: DateTime.now().toISO(), + updated: DateTime.now().toISO(), + }); + + await mswDB.add('domains', domain, mockState); + + queueEvents({ + event: { + action: 'domain_create', + entity: { + id: domain.id, + label: domain.domain, + type: 'domain', + url: `/v4/domains/${domain.id}`, + }, + }, + mockState, + sequence: [{ status: 'notification' }], + }); + + return makeResponse(domain); + } + ), +]; + +export const deleteDomains = (mockState: MockState) => [ + http.delete( + '*/v4/domains/:id', + async ({ params }): Promise> => { + const id = Number(params.id); + const domain = await mswDB.get('domains', id); + + if (!domain) { + return makeNotFoundResponse(); + } + + await mswDB.delete('domains', id, mockState); + + queueEvents({ + event: { + action: 'domain_delete', + entity: { + id: domain.id, + label: domain.domain, + type: 'domain', + url: `/v4/domains/${domain.id}`, + }, + }, + mockState, + sequence: [{ status: 'notification' }], + }); + + return makeResponse({}); + } + ), +]; diff --git a/packages/manager/src/mocks/presets/crud/seeds/domains.ts b/packages/manager/src/mocks/presets/crud/seeds/domains.ts new file mode 100644 index 00000000000..989aa240f83 --- /dev/null +++ b/packages/manager/src/mocks/presets/crud/seeds/domains.ts @@ -0,0 +1,32 @@ +import { getSeedsCountMap } from 'src/dev-tools/utils'; +import { domainFactory } from 'src/factories'; +import { mswDB } from 'src/mocks/indexedDB'; +import { seedWithUniqueIds } from 'src/mocks/presets/crud/seeds/utils'; + +import type { MockSeeder, MockState } from 'src/mocks/types'; + +export const domainSeeder: MockSeeder = { + canUpdateCount: true, + desc: 'Domains Seeds', + group: { id: 'Domains' }, + id: 'domains:crud', + label: 'Domains', + + seeder: async (mockState: MockState) => { + const seedsCountMap = getSeedsCountMap(); + const count = seedsCountMap[domainSeeder.id] ?? 0; + const domainSeeds = seedWithUniqueIds<'domains'>({ + dbEntities: await mswDB.getAll('domains'), + seedEntities: domainFactory.buildList(count), + }); + + const updatedMockState = { + ...mockState, + domains: mockState.domains.concat(domainSeeds), + }; + + await mswDB.saveStore(updatedMockState, 'seedState'); + + return updatedMockState; + }, +}; diff --git a/packages/manager/src/mocks/presets/crud/seeds/index.ts b/packages/manager/src/mocks/presets/crud/seeds/index.ts index f00084e08bc..3aab667a0e3 100644 --- a/packages/manager/src/mocks/presets/crud/seeds/index.ts +++ b/packages/manager/src/mocks/presets/crud/seeds/index.ts @@ -1,9 +1,11 @@ +import { domainSeeder } from './domains'; import { linodesSeeder } from './linodes'; import { placementGroupSeeder } from './placementGroups'; import { supportTicketsSeeder } from './supportTickets'; import { volumesSeeder } from './volumes'; export const dbSeeders = [ + domainSeeder, linodesSeeder, placementGroupSeeder, supportTicketsSeeder, diff --git a/packages/manager/src/mocks/presets/crud/seeds/utils.ts b/packages/manager/src/mocks/presets/crud/seeds/utils.ts index 5b77c2d2781..8fe4f536c89 100644 --- a/packages/manager/src/mocks/presets/crud/seeds/utils.ts +++ b/packages/manager/src/mocks/presets/crud/seeds/utils.ts @@ -13,6 +13,9 @@ import type { MockSeeder, MockState } from 'src/mocks/types'; */ export const removeSeeds = async (seederId: MockSeeder['id']) => { switch (seederId) { + case 'domains:crud': + await mswDB.deleteAll('domains', mockState, 'seedState'); + break; case 'linodes:crud': await mswDB.deleteAll('linodes', mockState, 'seedState'); await mswDB.deleteAll('linodeConfigs', mockState, 'seedState'); diff --git a/packages/manager/src/mocks/types.ts b/packages/manager/src/mocks/types.ts index 5b01ed9834e..7597a52cd2f 100644 --- a/packages/manager/src/mocks/types.ts +++ b/packages/manager/src/mocks/types.ts @@ -1,5 +1,7 @@ import type { Config, + Domain, + DomainRecord, Event, Firewall, Linode, @@ -79,9 +81,15 @@ export interface MockPresetExtra extends MockPresetBase { * Mock Preset Crud */ export type MockPresetCrudGroup = { - id: 'Linodes' | 'Placement Groups' | 'Support Tickets' | 'Volumes'; + id: + | 'Domains' + | 'Linodes' + | 'Placement Groups' + | 'Support Tickets' + | 'Volumes'; }; export type MockPresetCrudId = + | 'domains:crud' | 'linodes:crud' | 'placement-groups:crud' | 'support-tickets:crud' @@ -98,6 +106,8 @@ export type MockHandler = (mockState: MockState) => HttpHandler[]; * Stateful data shared among mocks. */ export interface MockState { + domainRecords: DomainRecord[]; + domains: Domain[]; eventQueue: Event[]; firewalls: Firewall[]; linodeConfigs: [number, Config][]; diff --git a/packages/manager/src/mocks/utilities/events.ts b/packages/manager/src/mocks/utilities/events.ts index 5c8b24a4286..f5c7bf4973d 100644 --- a/packages/manager/src/mocks/utilities/events.ts +++ b/packages/manager/src/mocks/utilities/events.ts @@ -28,7 +28,7 @@ const delay = (ms: number) => new Promise((resolve) => setTimeout(resolve, ms)); export const queueEvents = (props: QueuedEvents): Promise => { const { event, mockState, sequence } = props; - const initialDelay = 7500; + const initialDelay = 2500; const progressDelay = 10_000; let accumulatedDelay = 0; let lastEventWasProgress = false;