From df81d1dbc2e9e656f3b9eb701996c1d6499e1879 Mon Sep 17 00:00:00 2001 From: Angelo Ashmore Date: Mon, 16 Dec 2024 16:13:15 -1000 Subject: [PATCH 01/12] refactor: link types --- src/Migration.ts | 11 ++- src/lib/getLinkProperties.ts | 25 ++++++ src/lib/isValue.ts | 46 ++++------- src/lib/resolveMigrationDocumentData.ts | 6 +- src/types/migration/Asset.ts | 3 +- src/types/migration/ContentRelationship.ts | 19 ++--- src/types/model/contentRelationship.ts | 1 - src/types/value/contentRelationship.ts | 11 ++- src/types/value/link.ts | 76 +++++++++++-------- src/types/value/linkToMedia.ts | 13 ++-- ...ate-patch-contentRelationship.test.ts.snap | 14 ++-- ...ent-migrate-patch-linkToMedia.test.ts.snap | 14 ++-- test/helpers-asLink.test.ts | 22 ++---- test/helpers-asLinkAttrs.test.ts | 24 +----- test/helpers-isFilled.test.ts | 18 ++--- .../customType-contentRelationship.types.ts | 12 --- .../types/fields-contentRelationship.types.ts | 34 ++------- test/types/fields-link.types.ts | 72 ++++++------------ test/types/fields-linkToMedia.types.ts | 28 +++---- 19 files changed, 192 insertions(+), 257 deletions(-) create mode 100644 src/lib/getLinkProperties.ts diff --git a/src/Migration.ts b/src/Migration.ts index 53ab719a..89125878 100644 --- a/src/Migration.ts +++ b/src/Migration.ts @@ -1,4 +1,5 @@ import * as is from "./lib/isValue" +import { getOptionalLinkProperties } from "./lib/getLinkProperties" import { validateAssetMetadata } from "./lib/validateAssetMetadata" import type { Asset } from "./types/api/asset/asset" @@ -404,28 +405,32 @@ export class Migration { */ #migratePrismicDocumentData(input: unknown): unknown { if (is.filledContentRelationship(input)) { + const optionalLinkProperties = getOptionalLinkProperties(input) + if (input.isBroken) { return { + ...optionalLinkProperties, link_type: LinkType.Document, // ID needs to be 16 characters long to be considered valid by the API id: "_____broken_____", isBroken: true, - text: input.text, } } return { + ...optionalLinkProperties, link_type: LinkType.Document, id: () => this._getByOriginalID(input.id), - text: input.text, } } if (is.filledLinkToMedia(input)) { + const optionalLinkProperties = getOptionalLinkProperties(input) + return { + ...optionalLinkProperties, link_type: LinkType.Media, id: this.createAsset(input), - text: input.text, } } diff --git a/src/lib/getLinkProperties.ts b/src/lib/getLinkProperties.ts new file mode 100644 index 00000000..6f7a9131 --- /dev/null +++ b/src/lib/getLinkProperties.ts @@ -0,0 +1,25 @@ +import type { FilledContentRelationshipField } from "../types/value/contentRelationship" +import type { OptionalLinkProperties } from "../types/value/link" +import type { FilledLinkToMediaField } from "../types/value/linkToMedia" + +/** + * Returns optional properties only available to link fields. Link fields can + * have the same shape as content relationship and link to media fields, + * requiring special treatment to extract link-specific properties. + * + * @param input - The content relationship or link to media field from which the + * link properties are extracted. + * + * @returns Optional link properties that `input` might have. + */ +export const getOptionalLinkProperties = ( + input: FilledContentRelationshipField | FilledLinkToMediaField, +): OptionalLinkProperties => { + const res: OptionalLinkProperties = {} + + if ("text" in input) { + res.text = input.text + } + + return res +} diff --git a/src/lib/isValue.ts b/src/lib/isValue.ts index c4b5bd11..ca2eded9 100644 --- a/src/lib/isValue.ts +++ b/src/lib/isValue.ts @@ -3,6 +3,7 @@ import type { FilledContentRelationshipField } from "../types/value/contentRelat import type { PrismicDocument } from "../types/value/document" import type { GroupField } from "../types/value/group" import type { ImageField } from "../types/value/image" +import type { LinkField } from "../types/value/link" import { LinkType } from "../types/value/link" import type { FilledLinkToMediaField } from "../types/value/linkToMedia" import { type RTImageNode, RichTextNodeType } from "../types/value/richText" @@ -33,23 +34,7 @@ type UnknownValue = export const filledLinkToMedia = ( value: UnknownValue, ): value is FilledLinkToMediaField => { - if (value && typeof value === "object" && !("version" in value)) { - if ( - "link_type" in value && - value.link_type === LinkType.Media && - "id" in value && - "name" in value && - "kind" in value && - "url" in value && - "size" in value - ) { - value - - return true - } - } - - return false + return filledLink(value) && value.link_type === LinkType.Media } /** @@ -135,6 +120,18 @@ export const rtImageNode = (value: UnknownValue): value is RTImageNode => { return false } +export const filledLink = ( + value: UnknownValue, +): value is LinkField => { + return ( + typeof value === "object" && + value !== null && + "link_type" in value && + typeof value.link_type === "string" && + value.link_type !== LinkType.Any + ) +} + /** * Checks if a value is a content relationship field. * @@ -148,20 +145,7 @@ export const rtImageNode = (value: UnknownValue): value is RTImageNode => { export const filledContentRelationship = ( value: UnknownValue, ): value is FilledContentRelationshipField => { - if (value && typeof value === "object" && !("version" in value)) { - if ( - "link_type" in value && - value.link_type === LinkType.Document && - "id" in value && - "type" in value && - "tags" in value && - "lang" in value - ) { - return true - } - } - - return false + return filledLink(value) && value.link_type === LinkType.Document } /** diff --git a/src/lib/resolveMigrationDocumentData.ts b/src/lib/resolveMigrationDocumentData.ts index dded02c9..67fdd0a2 100644 --- a/src/lib/resolveMigrationDocumentData.ts +++ b/src/lib/resolveMigrationDocumentData.ts @@ -38,7 +38,7 @@ export async function resolveMigrationContentRelationship( if (relation instanceof PrismicMigrationDocument) { return relation.document.id ? { link_type: LinkType.Document, id: relation.document.id } - : { link_type: LinkType.Document } + : { link_type: LinkType.Any } } if (relation) { @@ -57,7 +57,7 @@ export async function resolveMigrationContentRelationship( return { link_type: LinkType.Document, id: relation.id } } - return { link_type: LinkType.Document } + return { link_type: LinkType.Any } } /** @@ -167,7 +167,7 @@ export const resolveMigrationLinkToMedia = ( } } - return { link_type: LinkType.Media } + return { link_type: LinkType.Any } } /** diff --git a/src/types/migration/Asset.ts b/src/types/migration/Asset.ts index 386eb0ce..2eb7a1aa 100644 --- a/src/types/migration/Asset.ts +++ b/src/types/migration/Asset.ts @@ -1,6 +1,5 @@ import type { Asset } from "../api/asset/asset" import type { FilledImageFieldImage } from "../value/image" -import type { EmptyLinkField } from "../value/link" import type { LinkToMediaField } from "../value/linkToMedia" import { type RTImageNode } from "../value/richText" @@ -89,7 +88,7 @@ export type MigrationLinkToMedia = Pick< */ export type MigrationLinkToMediaField = | Pick, "link_type" | "id" | "text"> - | EmptyLinkField<"Media"> + | LinkToMediaField<"empty"> /** * A rich text image node in a migration. diff --git a/src/types/migration/ContentRelationship.ts b/src/types/migration/ContentRelationship.ts index 667836f6..5f5edc18 100644 --- a/src/types/migration/ContentRelationship.ts +++ b/src/types/migration/ContentRelationship.ts @@ -1,6 +1,8 @@ -import type { FilledContentRelationshipField } from "../value/contentRelationship" +import type { + ContentRelationshipField, + FilledContentRelationshipField, +} from "../value/contentRelationship" import type { PrismicDocument } from "../value/document" -import type { EmptyLinkField } from "../value/link" import type { PrismicMigrationDocument } from "./Document" @@ -13,12 +15,11 @@ export type MigrationContentRelationship< TDocuments extends PrismicDocument = PrismicDocument, > = | ValueOrThunk | undefined> - | (Pick & - Partial> & { - id: ValueOrThunk< - TDocuments | PrismicMigrationDocument | undefined - > - }) + | (Pick & { + id: ValueOrThunk< + TDocuments | PrismicMigrationDocument | undefined + > + }) /** * The minimum amount of information needed to represent a content relationship @@ -26,4 +27,4 @@ export type MigrationContentRelationship< */ export type MigrationContentRelationshipField = | Pick - | EmptyLinkField<"Document"> + | ContentRelationshipField diff --git a/src/types/model/contentRelationship.ts b/src/types/model/contentRelationship.ts index 4c542748..09f06294 100644 --- a/src/types/model/contentRelationship.ts +++ b/src/types/model/contentRelationship.ts @@ -18,6 +18,5 @@ export interface CustomTypeModelContentRelationshipField< select: typeof CustomTypeModelLinkSelectType.Document customtypes?: readonly CustomTypeIDs[] tags?: readonly Tags[] - allowText?: boolean } } diff --git a/src/types/value/contentRelationship.ts b/src/types/value/contentRelationship.ts index 18919c3a..9e92a270 100644 --- a/src/types/value/contentRelationship.ts +++ b/src/types/value/contentRelationship.ts @@ -1,7 +1,7 @@ import type { AnyRegularField, FieldState } from "./types" import type { GroupField } from "./group" -import type { EmptyLinkField, LinkType } from "./link" +import type { LinkType } from "./link" import type { SliceZone } from "./sliceZone" /** @@ -21,9 +21,13 @@ export type ContentRelationshipField< | unknown = unknown, State extends FieldState = FieldState, > = State extends "empty" - ? EmptyLinkField + ? EmptyContentRelationshipField : FilledContentRelationshipField +type EmptyContentRelationshipField = { + link_type: typeof LinkType.Any +} + /** * Links that refer to documents */ @@ -34,7 +38,7 @@ export interface FilledContentRelationshipField< | Record | unknown = unknown, > { - link_type: typeof LinkType.Document + link_type: "Document" id: string uid?: string type: TypeEnum @@ -44,5 +48,4 @@ export interface FilledContentRelationshipField< slug?: string isBroken?: boolean data?: DataInterface - text?: string } diff --git a/src/types/value/link.ts b/src/types/value/link.ts index 870004b1..87ebfecb 100644 --- a/src/types/value/link.ts +++ b/src/types/value/link.ts @@ -1,8 +1,8 @@ import type { AnyRegularField, FieldState } from "./types" -import type { ContentRelationshipField } from "./contentRelationship" +import type { FilledContentRelationshipField } from "./contentRelationship" import type { GroupField } from "./group" -import type { LinkToMediaField } from "./linkToMedia" +import type { FilledLinkToMediaField } from "./linkToMedia" import type { SliceZone } from "./sliceZone" /** @@ -15,35 +15,13 @@ export const LinkType = { Web: "Web", } as const -/** - * For link fields that haven't been filled - * - * @typeParam Type - The type of link. - */ -export type EmptyLinkField< - Type extends (typeof LinkType)[keyof typeof LinkType] = typeof LinkType.Any, -> = { - link_type: Type | string - text?: string -} - -/** - * Link that points to external website - */ -export interface FilledLinkToWebField { - link_type: typeof LinkType.Web - url: string - target?: string - text?: string -} - /** * A link field. * * @typeParam TypeEnum - Type API ID of the document. * @typeParam LangEnum - Language API ID of the document. - * @typeParam DataInterface - Data fields for the document (filled in via - * GraphQuery of `fetchLinks`). + * @typeParam DataInterface - Data fields for the document (filled via the + * `fetchLinks` or `graphQuery` query parameter). * @typeParam State - State of the field which determines its shape. */ export type LinkField< @@ -53,9 +31,45 @@ export type LinkField< | Record | unknown = unknown, State extends FieldState = FieldState, -> = State extends "empty" - ? EmptyLinkField +> = (State extends "empty" + ? EmptyLinkField : - | ContentRelationshipField - | FilledLinkToWebField - | LinkToMediaField + | FilledContentRelationshipField + | FilledLinkToMediaField + | FilledLinkToWebField) & + OptionalLinkProperties + +/** + * A link field that is not filled. + * + * @typeParam Type - Unused. An empty link field will always have a `link_type` + * of "Any". + */ +// This type needs `OptionalLinkProperties` because this type may be used on its own. +export type EmptyLinkField< + _Unused extends + (typeof LinkType)[keyof typeof LinkType] = typeof LinkType.Any, +> = { + link_type: typeof LinkType.Any +} & OptionalLinkProperties + +/** + * A link field pointing to a relative or absolute URL. + */ +// This type needs `OptionalLinkProperties` because this type may be used on its own. +export type FilledLinkToWebField = { + link_type: typeof LinkType.Web + url: string + target?: string +} & OptionalLinkProperties + +/** + * Optional properties available to link fields. It is used to augment existing + * link-like fields (like content relationship fields) with field-specific + * properties. + * + * @internal + */ +export type OptionalLinkProperties = { + text?: string +} diff --git a/src/types/value/linkToMedia.ts b/src/types/value/linkToMedia.ts index 85d1a92c..453a9c3b 100644 --- a/src/types/value/linkToMedia.ts +++ b/src/types/value/linkToMedia.ts @@ -1,6 +1,6 @@ import type { FieldState } from "./types" -import type { EmptyLinkField, LinkType } from "./link" +import type { LinkType } from "./link" /** * A link field that points to media. @@ -8,16 +8,19 @@ import type { EmptyLinkField, LinkType } from "./link" * @typeParam State - State of the field which determines its shape. */ export type LinkToMediaField = - State extends "empty" - ? EmptyLinkField - : FilledLinkToMediaField + State extends "empty" ? EmptyLinkToMediaField : FilledLinkToMediaField + +type EmptyLinkToMediaField = { + link_type: typeof LinkType.Any + text?: string +} /** * A link that points to media. */ export interface FilledLinkToMediaField { id: string - link_type: typeof LinkType.Media + link_type: "Media" name: string kind: string url: string diff --git a/test/__snapshots__/writeClient-migrate-patch-contentRelationship.test.ts.snap b/test/__snapshots__/writeClient-migrate-patch-contentRelationship.test.ts.snap index 337e410a..3b2282d6 100644 --- a/test/__snapshots__/writeClient-migrate-patch-contentRelationship.test.ts.snap +++ b/test/__snapshots__/writeClient-migrate-patch-contentRelationship.test.ts.snap @@ -698,7 +698,7 @@ exports[`patches content relationship fields > lazyOtherCreateMissingID > group "group": [ { "field": { - "link_type": "Document", + "link_type": "Any", }, }, ], @@ -713,18 +713,18 @@ exports[`patches content relationship fields > lazyOtherCreateMissingID > shared "items": [ { "field": { - "link_type": "Document", + "link_type": "Any", }, }, ], "primary": { "field": { - "link_type": "Document", + "link_type": "Any", }, "group": [ { "field": { - "link_type": "Document", + "link_type": "Any", }, }, ], @@ -746,13 +746,13 @@ exports[`patches content relationship fields > lazyOtherCreateMissingID > slice "items": [ { "field": { - "link_type": "Document", + "link_type": "Any", }, }, ], "primary": { "field": { - "link_type": "Document", + "link_type": "Any", }, }, "slice_label": "Vel", @@ -765,7 +765,7 @@ exports[`patches content relationship fields > lazyOtherCreateMissingID > slice exports[`patches content relationship fields > lazyOtherCreateMissingID > static zone 1`] = ` { "field": { - "link_type": "Document", + "link_type": "Any", }, } `; diff --git a/test/__snapshots__/writeClient-migrate-patch-linkToMedia.test.ts.snap b/test/__snapshots__/writeClient-migrate-patch-linkToMedia.test.ts.snap index 9275be7f..c394567e 100644 --- a/test/__snapshots__/writeClient-migrate-patch-linkToMedia.test.ts.snap +++ b/test/__snapshots__/writeClient-migrate-patch-linkToMedia.test.ts.snap @@ -299,7 +299,7 @@ exports[`patches link to media fields > empty > group 1`] = ` "group": [ { "field": { - "link_type": "Media", + "link_type": "Any", }, }, ], @@ -314,18 +314,18 @@ exports[`patches link to media fields > empty > shared slice 1`] = ` "items": [ { "field": { - "link_type": "Media", + "link_type": "Any", }, }, ], "primary": { "field": { - "link_type": "Media", + "link_type": "Any", }, "group": [ { "field": { - "link_type": "Media", + "link_type": "Any", }, }, ], @@ -347,13 +347,13 @@ exports[`patches link to media fields > empty > slice 1`] = ` "items": [ { "field": { - "link_type": "Media", + "link_type": "Any", }, }, ], "primary": { "field": { - "link_type": "Media", + "link_type": "Any", }, }, "slice_label": "Vel", @@ -366,7 +366,7 @@ exports[`patches link to media fields > empty > slice 1`] = ` exports[`patches link to media fields > empty > static zone 1`] = ` { "field": { - "link_type": "Media", + "link_type": "Any", }, } `; diff --git a/test/helpers-asLink.test.ts b/test/helpers-asLink.test.ts index 80264fd7..cf8686b7 100644 --- a/test/helpers-asLink.test.ts +++ b/test/helpers-asLink.test.ts @@ -10,22 +10,6 @@ it("returns null for nullish inputs", () => { expect(asLink(undefined, linkResolver)).toBeNull() }) -it("returns null when link to document field is empty", () => { - const field = { - link_type: LinkType.Document, - } - - expect(asLink(field, linkResolver)).toBeNull() -}) - -it("returns null when link to media field is empty", () => { - const field = { - link_type: LinkType.Media, - } - - expect(asLink(field, linkResolver)).toBeNull() -}) - it("returns null when link field is empty", () => { const field = { link_type: LinkType.Any, @@ -102,8 +86,11 @@ it("resolves a link to document field with route resolver", () => { it("returns null when given a document field and linkResolver is not provided ", () => { const field = { - id: "XvoFFREAAM0WGBng", link_type: LinkType.Document, + id: "XvoFFREAAM0WGBng", + lang: "fr-fr", + tags: [], + type: "foo", } expect(asLink(field)).toBeNull() @@ -121,6 +108,7 @@ it("resolves a link to web field", () => { it("resolves a link to media field", () => { const field = { link_type: LinkType.Media, + id: "foo", name: "test.jpg", kind: "image", url: "https://prismic.io", diff --git a/test/helpers-asLinkAttrs.test.ts b/test/helpers-asLinkAttrs.test.ts index a37ee933..a8c21b87 100644 --- a/test/helpers-asLinkAttrs.test.ts +++ b/test/helpers-asLinkAttrs.test.ts @@ -1,32 +1,14 @@ import { expect, it, vi } from "vitest" -import { asLinkAttrs } from "../src" +import { LinkType, asLinkAttrs } from "../src" it("returns empty object for nullish inputs", () => { expect(asLinkAttrs(null)).toEqual({}) expect(asLinkAttrs(undefined)).toEqual({}) }) -it("returns empty object when link field is empty", (ctx) => { - const field = ctx.mock.value.link({ type: "Any", state: "empty" }) - - expect(asLinkAttrs(field)).toEqual({}) -}) - -it("returns empty object when link to document field is empty", (ctx) => { - const field = ctx.mock.value.link({ type: "Document", state: "empty" }) - - expect(asLinkAttrs(field)).toEqual({}) -}) - -it("returns empty object when link to media field is empty", (ctx) => { - const field = ctx.mock.value.link({ type: "Media", state: "empty" }) - - expect(asLinkAttrs(field)).toEqual({}) -}) - -it("returns empty object when link to web field is empty", (ctx) => { - const field = ctx.mock.value.link({ type: "Web", state: "empty" }) +it("returns empty object when link field is empty", () => { + const field = { link_type: LinkType.Any } expect(asLinkAttrs(field)).toEqual({}) }) diff --git a/test/helpers-isFilled.test.ts b/test/helpers-isFilled.test.ts index 4f7b7fcf..79958475 100644 --- a/test/helpers-isFilled.test.ts +++ b/test/helpers-isFilled.test.ts @@ -3,7 +3,9 @@ import { expect, it } from "vitest" import { documentFixture } from "./__fixtures__/document" import type { GroupField, SliceZone } from "../src" -import { isFilled } from "../src" +import { LinkType, isFilled } from "../src" + +const emptyLink = { link_type: LinkType.Any } it("color", (ctx) => { expect(isFilled.color(null)).toBe(false) @@ -14,13 +16,7 @@ it("color", (ctx) => { it("content relationship", (ctx) => { expect(isFilled.contentRelationship(null)).toBe(false) expect(isFilled.contentRelationship(undefined)).toBe(false) - expect( - isFilled.contentRelationship( - ctx.mock.value.contentRelationship({ - state: "empty", - }), - ), - ).toBe(false) + expect(isFilled.contentRelationship(emptyLink)).toBe(false) expect( isFilled.contentRelationship( ctx.mock.value.contentRelationship({ @@ -104,16 +100,14 @@ it("key text", (ctx) => { it("link", (ctx) => { expect(isFilled.link(null)).toBe(false) expect(isFilled.link(undefined)).toBe(false) - expect(isFilled.link(ctx.mock.value.link({ state: "empty" }))).toBe(false) + expect(isFilled.link(emptyLink)).toBe(false) expect(isFilled.link(ctx.mock.value.link({ state: "filled" }))).toBe(true) }) it("link to media", (ctx) => { expect(isFilled.linkToMedia(null)).toBe(false) expect(isFilled.linkToMedia(undefined)).toBe(false) - expect( - isFilled.linkToMedia(ctx.mock.value.linkToMedia({ state: "empty" })), - ).toBe(false) + expect(isFilled.linkToMedia(emptyLink)).toBe(false) expect( isFilled.linkToMedia(ctx.mock.value.linkToMedia({ state: "filled" })), ).toBe(true) diff --git a/test/types/customType-contentRelationship.types.ts b/test/types/customType-contentRelationship.types.ts index 0648933e..2789c2be 100644 --- a/test/types/customType-contentRelationship.types.ts +++ b/test/types/customType-contentRelationship.types.ts @@ -96,18 +96,6 @@ expectType>({ }, }) -/** - * Supports optional `allowText` property. - */ -expectType>({ - type: prismic.CustomTypeModelFieldType.Link, - config: { - label: "string", - select: prismic.CustomTypeModelLinkSelectType.Document, - allowText: true, - }, -}) - /** * `@prismicio/types` extends `@prismicio/types-internal` */ diff --git a/test/types/fields-contentRelationship.types.ts b/test/types/fields-contentRelationship.types.ts index 71f455f3..91e3b2c4 100644 --- a/test/types/fields-contentRelationship.types.ts +++ b/test/types/fields-contentRelationship.types.ts @@ -32,7 +32,6 @@ expectType({ slug: "string", isBroken: true, data: undefined, - text: "string", }) expectType>({ link_type: prismic.LinkType.Document, @@ -45,10 +44,9 @@ expectType>({ slug: "string", isBroken: true, data: undefined, - text: "string", }) expectType>({ - link_type: prismic.LinkType.Document, + link_type: prismic.LinkType.Any, // @ts-expect-error - Empty fields cannot contain a filled value. id: "string", uid: "string", @@ -59,43 +57,21 @@ expectType>({ slug: "string", isBroken: true, data: undefined, - text: "string", }) /** * Empty state. */ expectType({ - link_type: prismic.LinkType.Document, + link_type: prismic.LinkType.Any, }) expectType>({ - link_type: prismic.LinkType.Document, + link_type: prismic.LinkType.Any, }) -expectType>( +expectType>({ // @ts-expect-error - Filled fields cannot contain an empty value. - { - link_type: prismic.LinkType.Document, - }, -) - -/** - * Empty state with text. - */ -expectType({ - link_type: prismic.LinkType.Document, - text: "string", + link_type: prismic.LinkType.Any, }) -expectType>({ - link_type: prismic.LinkType.Document, - text: "string", -}) -expectType>( - // @ts-expect-error - Filled fields cannot contain an empty value. - { - link_type: prismic.LinkType.Document, - text: "string", - }, -) /** * Supports custom document type. diff --git a/test/types/fields-link.types.ts b/test/types/fields-link.types.ts index 2f8a3a5d..e444a914 100644 --- a/test/types/fields-link.types.ts +++ b/test/types/fields-link.types.ts @@ -36,14 +36,12 @@ expectType( /** * Filled state. */ -// Web link expectType({ link_type: prismic.LinkType.Web, url: "string", target: "string", text: "string", }) -// Content relationship link expectType({ link_type: prismic.LinkType.Document, id: "string", @@ -57,9 +55,9 @@ expectType({ data: undefined, text: "string", }) -// Media link expectType({ link_type: prismic.LinkType.Media, + id: "string", name: "string", kind: "string", url: "string", @@ -75,8 +73,8 @@ expectType>({ text: "string", }) expectType>({ + // @ts-expect-error - Empty fields cannot contain a filled link type. link_type: prismic.LinkType.Web, - // @ts-expect-error - Empty fields cannot contain a filled value. url: "string", target: "string", text: "string", @@ -92,70 +90,50 @@ expectType>({ // @ts-expect-error - Filled fields cannot contain an empty value. link_type: prismic.LinkType.Any, }) -expectType({ - link_type: prismic.LinkType.Web, -}) -expectType>( - // @ts-expect-error - Filled fields cannot contain an empty value. +expectType( + // @ts-expect-error - Filled fields must contain properties. { link_type: prismic.LinkType.Web, }, ) -expectType({ - link_type: prismic.LinkType.Document, -}) expectType>( - // @ts-expect-error - Filled fields cannot contain an empty value. + // @ts-expect-error - Filled fields must contain properties. { - link_type: prismic.LinkType.Document, + link_type: prismic.LinkType.Web, }, ) -expectType({ - link_type: prismic.LinkType.Media, -}) -expectType>( - // @ts-expect-error - Filled fields cannot contain an empty value. +expectType( + // @ts-expect-error - Filled fields must contain properties. { - link_type: prismic.LinkType.Media, + link_type: prismic.LinkType.Document, }, ) - -/** - * Empty state with text. - */ -expectType>({ - link_type: prismic.LinkType.Web, - text: "string", -}) expectType>( - // @ts-expect-error - Filled fields cannot contain an empty value. + // @ts-expect-error - Filled fields must contain properties. { - link_type: prismic.LinkType.Web, - text: "string", + link_type: prismic.LinkType.Document, }, ) -expectType>({ - link_type: prismic.LinkType.Document, - text: "string", -}) -expectType>( - // @ts-expect-error - Filled fields cannot contain an empty value. +expectType( + // @ts-expect-error - Filled fields must contain properties. { - link_type: prismic.LinkType.Document, - text: "string", + link_type: prismic.LinkType.Media, }, ) -expectType>({ - link_type: prismic.LinkType.Media, - text: "string", -}) expectType>( - // @ts-expect-error - Filled fields cannot contain an empty value. + // @ts-expect-error - Filled fields must contain properties. { link_type: prismic.LinkType.Media, - text: "string", }, ) + +/** + * Empty state with text. + */ +expectType({ + link_type: prismic.LinkType.Any, + text: "string", +}) expectType>({ link_type: prismic.LinkType.Any, text: "string", @@ -176,10 +154,10 @@ expectType>({ tags: [], lang: "string", }) -// @ts-expect-error - Document type must match the given type. expectType>({ link_type: prismic.LinkType.Document, id: "string", + // @ts-expect-error - Document type must match the given type. type: "string", tags: [], lang: "string", @@ -195,12 +173,12 @@ expectType>({ tags: [], lang: "fr-fr", }) -// @ts-expect-error - Document language must match the given type. expectType>({ link_type: prismic.LinkType.Document, id: "string", type: "string", tags: [], + // @ts-expect-error - Document language must match the given type. lang: "string", }) diff --git a/test/types/fields-linkToMedia.types.ts b/test/types/fields-linkToMedia.types.ts index c5b63fbf..671d6b66 100644 --- a/test/types/fields-linkToMedia.types.ts +++ b/test/types/fields-linkToMedia.types.ts @@ -44,7 +44,7 @@ expectType>({ text: "string", }) expectType>({ - link_type: prismic.LinkType.Media, + link_type: prismic.LinkType.Any, // @ts-expect-error - Empty fields cannot contain a filled value. id: "string", name: "string", @@ -60,33 +60,29 @@ expectType>({ * Empty state. */ expectType({ - link_type: prismic.LinkType.Media, + link_type: prismic.LinkType.Any, }) expectType>({ - link_type: prismic.LinkType.Media, + link_type: prismic.LinkType.Any, }) -expectType>( +expectType>({ // @ts-expect-error - Filled fields cannot contain an empty value. - { - link_type: prismic.LinkType.Media, - }, -) + link_type: prismic.LinkType.Any, +}) /** * Empty state with text. */ expectType({ - link_type: prismic.LinkType.Media, + link_type: prismic.LinkType.Any, text: "string", }) expectType>({ - link_type: prismic.LinkType.Media, + link_type: prismic.LinkType.Any, text: "string", }) -expectType>( +expectType>({ // @ts-expect-error - Filled fields cannot contain an empty value. - { - link_type: prismic.LinkType.Media, - text: "string", - }, -) + link_type: prismic.LinkType.Any, + text: "string", +}) From 0405e7c89d3fc358e6875b96e0f0996c5a3e4bd4 Mon Sep 17 00:00:00 2001 From: Angelo Ashmore Date: Mon, 16 Dec 2024 16:30:24 -1000 Subject: [PATCH 02/12] refactor: clean up after self-review --- src/lib/isValue.ts | 46 +++++++++++++++++--------- src/types/value/contentRelationship.ts | 3 +- src/types/value/link.ts | 8 ++--- src/types/value/linkToMedia.ts | 4 +-- 4 files changed, 37 insertions(+), 24 deletions(-) diff --git a/src/lib/isValue.ts b/src/lib/isValue.ts index ca2eded9..c4b5bd11 100644 --- a/src/lib/isValue.ts +++ b/src/lib/isValue.ts @@ -3,7 +3,6 @@ import type { FilledContentRelationshipField } from "../types/value/contentRelat import type { PrismicDocument } from "../types/value/document" import type { GroupField } from "../types/value/group" import type { ImageField } from "../types/value/image" -import type { LinkField } from "../types/value/link" import { LinkType } from "../types/value/link" import type { FilledLinkToMediaField } from "../types/value/linkToMedia" import { type RTImageNode, RichTextNodeType } from "../types/value/richText" @@ -34,7 +33,23 @@ type UnknownValue = export const filledLinkToMedia = ( value: UnknownValue, ): value is FilledLinkToMediaField => { - return filledLink(value) && value.link_type === LinkType.Media + if (value && typeof value === "object" && !("version" in value)) { + if ( + "link_type" in value && + value.link_type === LinkType.Media && + "id" in value && + "name" in value && + "kind" in value && + "url" in value && + "size" in value + ) { + value + + return true + } + } + + return false } /** @@ -120,18 +135,6 @@ export const rtImageNode = (value: UnknownValue): value is RTImageNode => { return false } -export const filledLink = ( - value: UnknownValue, -): value is LinkField => { - return ( - typeof value === "object" && - value !== null && - "link_type" in value && - typeof value.link_type === "string" && - value.link_type !== LinkType.Any - ) -} - /** * Checks if a value is a content relationship field. * @@ -145,7 +148,20 @@ export const filledLink = ( export const filledContentRelationship = ( value: UnknownValue, ): value is FilledContentRelationshipField => { - return filledLink(value) && value.link_type === LinkType.Document + if (value && typeof value === "object" && !("version" in value)) { + if ( + "link_type" in value && + value.link_type === LinkType.Document && + "id" in value && + "type" in value && + "tags" in value && + "lang" in value + ) { + return true + } + } + + return false } /** diff --git a/src/types/value/contentRelationship.ts b/src/types/value/contentRelationship.ts index 9e92a270..232fc443 100644 --- a/src/types/value/contentRelationship.ts +++ b/src/types/value/contentRelationship.ts @@ -1,7 +1,6 @@ import type { AnyRegularField, FieldState } from "./types" import type { GroupField } from "./group" -import type { LinkType } from "./link" import type { SliceZone } from "./sliceZone" /** @@ -25,7 +24,7 @@ export type ContentRelationshipField< : FilledContentRelationshipField type EmptyContentRelationshipField = { - link_type: typeof LinkType.Any + link_type: "Any" } /** diff --git a/src/types/value/link.ts b/src/types/value/link.ts index 87ebfecb..0da5a35c 100644 --- a/src/types/value/link.ts +++ b/src/types/value/link.ts @@ -42,15 +42,15 @@ export type LinkField< /** * A link field that is not filled. * - * @typeParam Type - Unused. An empty link field will always have a `link_type` - * of "Any". + * @typeParam _Unused - THIS PARAMETER IS NOT USED. If you are passing a type, + * **please remove it**. */ // This type needs `OptionalLinkProperties` because this type may be used on its own. export type EmptyLinkField< _Unused extends (typeof LinkType)[keyof typeof LinkType] = typeof LinkType.Any, > = { - link_type: typeof LinkType.Any + link_type: "Any" } & OptionalLinkProperties /** @@ -58,7 +58,7 @@ export type EmptyLinkField< */ // This type needs `OptionalLinkProperties` because this type may be used on its own. export type FilledLinkToWebField = { - link_type: typeof LinkType.Web + link_type: "Web" url: string target?: string } & OptionalLinkProperties diff --git a/src/types/value/linkToMedia.ts b/src/types/value/linkToMedia.ts index 453a9c3b..281989de 100644 --- a/src/types/value/linkToMedia.ts +++ b/src/types/value/linkToMedia.ts @@ -1,7 +1,5 @@ import type { FieldState } from "./types" -import type { LinkType } from "./link" - /** * A link field that points to media. * @@ -11,7 +9,7 @@ export type LinkToMediaField = State extends "empty" ? EmptyLinkToMediaField : FilledLinkToMediaField type EmptyLinkToMediaField = { - link_type: typeof LinkType.Any + link_type: "Any" text?: string } From 21d56d50ac9a599328dbf218db61022ff1858627 Mon Sep 17 00:00:00 2001 From: Angelo Ashmore Date: Mon, 16 Dec 2024 16:46:10 -1000 Subject: [PATCH 03/12] feat: support link variants --- src/types/model/link.ts | 1 + src/types/value/link.ts | 1 + test/types/customType-link.types.ts | 10 ++++++++++ test/types/fields-link.types.ts | 22 ++++++++++++++++++++++ 4 files changed, 34 insertions(+) diff --git a/src/types/model/link.ts b/src/types/model/link.ts index 4cb4c5fd..b4e14081 100644 --- a/src/types/model/link.ts +++ b/src/types/model/link.ts @@ -16,6 +16,7 @@ export interface CustomTypeModelLinkField { allowText?: boolean allowTargetBlank?: boolean repeat?: boolean + variants?: string[] } } diff --git a/src/types/value/link.ts b/src/types/value/link.ts index 0da5a35c..fb01b30c 100644 --- a/src/types/value/link.ts +++ b/src/types/value/link.ts @@ -71,5 +71,6 @@ export type FilledLinkToWebField = { * @internal */ export type OptionalLinkProperties = { + variant?: string text?: string } diff --git a/test/types/customType-link.types.ts b/test/types/customType-link.types.ts index 32ca8a7f..a5af283d 100644 --- a/test/types/customType-link.types.ts +++ b/test/types/customType-link.types.ts @@ -82,6 +82,16 @@ expectType({ }, }) +/** + * Supports optional `variants` property. + */ +expectType({ + type: prismic.CustomTypeModelFieldType.Link, + config: { + variants: ["string"], + }, +}) + /** * `@prismicio/types` extends `@prismicio/types-internal` */ diff --git a/test/types/fields-link.types.ts b/test/types/fields-link.types.ts index e444a914..5e263b30 100644 --- a/test/types/fields-link.types.ts +++ b/test/types/fields-link.types.ts @@ -41,6 +41,7 @@ expectType({ url: "string", target: "string", text: "string", + variant: "string", }) expectType({ link_type: prismic.LinkType.Document, @@ -54,6 +55,7 @@ expectType({ isBroken: true, data: undefined, text: "string", + variant: "string", }) expectType({ link_type: prismic.LinkType.Media, @@ -65,12 +67,14 @@ expectType({ height: "string", width: "string", text: "string", + variant: "string", }) expectType>({ link_type: prismic.LinkType.Web, url: "string", target: "string", text: "string", + variant: "string", }) expectType>({ // @ts-expect-error - Empty fields cannot contain a filled link type. @@ -78,6 +82,7 @@ expectType>({ url: "string", target: "string", text: "string", + variant: "string", }) /** @@ -144,6 +149,23 @@ expectType>({ text: "string", }) +/** + * Empty state with variant. + */ +expectType({ + link_type: prismic.LinkType.Any, + variant: "string", +}) +expectType>({ + link_type: prismic.LinkType.Any, + variant: "string", +}) +expectType>({ + // @ts-expect-error - Filled fields cannot contain an empty value. + link_type: prismic.LinkType.Any, + variant: "string", +}) + /** * Supports custom document type for document links. */ From bd19dce2f7c85b090af4ac0a68ae3ea32a8275ee Mon Sep 17 00:00:00 2001 From: Angelo Ashmore Date: Mon, 16 Dec 2024 16:49:28 -1000 Subject: [PATCH 04/12] test: remove unnecessary test --- test/types/fields-link.types.ts | 5 ----- 1 file changed, 5 deletions(-) diff --git a/test/types/fields-link.types.ts b/test/types/fields-link.types.ts index 5e263b30..b8827548 100644 --- a/test/types/fields-link.types.ts +++ b/test/types/fields-link.types.ts @@ -160,11 +160,6 @@ expectType>({ link_type: prismic.LinkType.Any, variant: "string", }) -expectType>({ - // @ts-expect-error - Filled fields cannot contain an empty value. - link_type: prismic.LinkType.Any, - variant: "string", -}) /** * Supports custom document type for document links. From 8b08803d92ebc2cd317ffe08ef1a35e35a99114e Mon Sep 17 00:00:00 2001 From: Angelo Ashmore Date: Tue, 17 Dec 2024 09:20:17 -1000 Subject: [PATCH 05/12] refactor: fix name of `getOptionalLinkProperties` file --- src/Migration.ts | 2 +- src/lib/{getLinkProperties.ts => getOptionalLinkProperties.ts} | 0 2 files changed, 1 insertion(+), 1 deletion(-) rename src/lib/{getLinkProperties.ts => getOptionalLinkProperties.ts} (100%) diff --git a/src/Migration.ts b/src/Migration.ts index 89125878..50142440 100644 --- a/src/Migration.ts +++ b/src/Migration.ts @@ -1,5 +1,5 @@ import * as is from "./lib/isValue" -import { getOptionalLinkProperties } from "./lib/getLinkProperties" +import { getOptionalLinkProperties } from "./lib/getOptionalLinkProperties" import { validateAssetMetadata } from "./lib/validateAssetMetadata" import type { Asset } from "./types/api/asset/asset" diff --git a/src/lib/getLinkProperties.ts b/src/lib/getOptionalLinkProperties.ts similarity index 100% rename from src/lib/getLinkProperties.ts rename to src/lib/getOptionalLinkProperties.ts From 40731744496e9af85fde54a751e7bfc417228640 Mon Sep 17 00:00:00 2001 From: Angelo Ashmore Date: Tue, 17 Dec 2024 12:12:16 -1000 Subject: [PATCH 06/12] fix(migration): only include optional link properties in links --- src/lib/getOptionalLinkProperties.ts | 6 +-- src/lib/isMigrationValue.ts | 7 +++- src/lib/isValue.ts | 15 +++++++- src/lib/resolveMigrationDocumentData.ts | 43 ++++++++++++++++------ src/types/migration/Asset.ts | 15 ++++---- src/types/migration/ContentRelationship.ts | 3 +- src/types/migration/Link.ts | 6 +++ src/types/value/contentRelationship.ts | 2 +- src/types/value/link.ts | 31 ++++++++++++---- src/types/value/richText.ts | 14 ++----- 10 files changed, 95 insertions(+), 47 deletions(-) create mode 100644 src/types/migration/Link.ts diff --git a/src/lib/getOptionalLinkProperties.ts b/src/lib/getOptionalLinkProperties.ts index 6f7a9131..de83ea2a 100644 --- a/src/lib/getOptionalLinkProperties.ts +++ b/src/lib/getOptionalLinkProperties.ts @@ -1,6 +1,4 @@ -import type { FilledContentRelationshipField } from "../types/value/contentRelationship" import type { OptionalLinkProperties } from "../types/value/link" -import type { FilledLinkToMediaField } from "../types/value/linkToMedia" /** * Returns optional properties only available to link fields. Link fields can @@ -8,12 +6,12 @@ import type { FilledLinkToMediaField } from "../types/value/linkToMedia" * requiring special treatment to extract link-specific properties. * * @param input - The content relationship or link to media field from which the - * link properties are extracted. + * link properties will be extracted. * * @returns Optional link properties that `input` might have. */ export const getOptionalLinkProperties = ( - input: FilledContentRelationshipField | FilledLinkToMediaField, + input: OptionalLinkProperties, ): OptionalLinkProperties => { const res: OptionalLinkProperties = {} diff --git a/src/lib/isMigrationValue.ts b/src/lib/isMigrationValue.ts index 28a442e1..72613978 100644 --- a/src/lib/isMigrationValue.ts +++ b/src/lib/isMigrationValue.ts @@ -12,6 +12,7 @@ import { import type { PrismicDocument } from "../types/value/document" import type { GroupField } from "../types/value/group" import { LinkType } from "../types/value/link" +import type { OptionalLinkProperties } from "../types/value/link" import { RichTextNodeType } from "../types/value/richText" import type { SliceZone } from "../types/value/sliceZone" import type { AnyRegularField } from "../types/value/types" @@ -32,6 +33,10 @@ type UnknownValue = /** * Checks if a value is a migration content relationship field. * + * @remarks + * `OptionalLinkProperties` is included because `MigrationContentRelationship` + * may be a link field, not strictly a content relationship field. + * * @param value - Value to check. * * @returns `true` if `value` is a migration content relationship field, `false` @@ -42,7 +47,7 @@ type UnknownValue = */ export const contentRelationship = ( value: UnknownValue, -): value is MigrationContentRelationship => { +): value is MigrationContentRelationship & OptionalLinkProperties => { return ( value instanceof PrismicMigrationDocument || is.prismicDocument(value) || diff --git a/src/lib/isValue.ts b/src/lib/isValue.ts index c4b5bd11..f0ac7989 100644 --- a/src/lib/isValue.ts +++ b/src/lib/isValue.ts @@ -4,6 +4,7 @@ import type { PrismicDocument } from "../types/value/document" import type { GroupField } from "../types/value/group" import type { ImageField } from "../types/value/image" import { LinkType } from "../types/value/link" +import type { OptionalLinkProperties } from "../types/value/link" import type { FilledLinkToMediaField } from "../types/value/linkToMedia" import { type RTImageNode, RichTextNodeType } from "../types/value/richText" import type { SliceZone } from "../types/value/sliceZone" @@ -23,6 +24,11 @@ type UnknownValue = /** * Checks if a value is a link to media field. * + * @remarks + * The return value includes `OptionalLinkProperties` because + * `FilledLinkToMediaField` may be a link field, not strictly a content + * relationship field. + * * @param value - Value to check. * * @returns `true` if `value` is a link to media field, `false` otherwise. @@ -32,7 +38,7 @@ type UnknownValue = */ export const filledLinkToMedia = ( value: UnknownValue, -): value is FilledLinkToMediaField => { +): value is FilledLinkToMediaField & OptionalLinkProperties => { if (value && typeof value === "object" && !("version" in value)) { if ( "link_type" in value && @@ -138,6 +144,11 @@ export const rtImageNode = (value: UnknownValue): value is RTImageNode => { /** * Checks if a value is a content relationship field. * + * @remarks + * The return value includes `OptionalLinkProperties` because + * `FilledContentRelationshipField` may be a link field, not strictly a content + * relationship field. + * * @param value - Value to check. * * @returns `true` if `value` is a content relationship, `false` otherwise. @@ -147,7 +158,7 @@ export const rtImageNode = (value: UnknownValue): value is RTImageNode => { */ export const filledContentRelationship = ( value: UnknownValue, -): value is FilledContentRelationshipField => { +): value is FilledContentRelationshipField & OptionalLinkProperties => { if (value && typeof value === "object" && !("version" in value)) { if ( "link_type" in value && diff --git a/src/lib/resolveMigrationDocumentData.ts b/src/lib/resolveMigrationDocumentData.ts index 67fdd0a2..8f6e8c29 100644 --- a/src/lib/resolveMigrationDocumentData.ts +++ b/src/lib/resolveMigrationDocumentData.ts @@ -10,8 +10,9 @@ import type { MigrationContentRelationshipField, } from "../types/migration/ContentRelationship" import { PrismicMigrationDocument } from "../types/migration/Document" +import type { MaybeLink } from "../types/migration/Link" import type { FilledImageFieldImage } from "../types/value/image" -import type { LinkField } from "../types/value/link" +import type { LinkField, OptionalLinkProperties } from "../types/value/link" import { LinkType } from "../types/value/link" import type { RTImageNode } from "../types/value/richText" import { RichTextNodeType } from "../types/value/richText" @@ -20,6 +21,7 @@ import * as isFilled from "../helpers/isFilled" import type { Migration } from "../Migration" import * as isMigration from "./isMigrationValue" +import { getOptionalLinkProperties } from "./getOptionalLinkProperties" /** * Resolves a migration content relationship to a content relationship field. @@ -29,8 +31,8 @@ import * as isMigration from "./isMigrationValue" * @returns Resolved content relationship field. */ export async function resolveMigrationContentRelationship( - relation: MigrationContentRelationship, -): Promise { + relation: MaybeLink, +): Promise { if (typeof relation === "function") { return resolveMigrationContentRelationship(await relation()) } @@ -41,23 +43,36 @@ export async function resolveMigrationContentRelationship( : { link_type: LinkType.Any } } + const optionalLinkProperties = + relation && "link_type" in relation + ? getOptionalLinkProperties(relation) + : undefined + if (relation) { if ( isMigration.contentRelationship(relation.id) || - typeof relation.id === "function" + typeof relation.id !== "string" ) { return { + ...optionalLinkProperties, ...(await resolveMigrationContentRelationship(relation.id)), - // TODO: Remove when link text PR is merged - // @ts-expect-error - Future-proofing for link text - text: relation.text, } } - return { link_type: LinkType.Document, id: relation.id } + // This is only called when resolveMigrationContentRelationship recursively + // calls itself from the statement above and the resolved content relation + // is a Prismic document value. + return { + ...optionalLinkProperties, + link_type: LinkType.Document, + id: relation.id, + } } - return { link_type: LinkType.Any } + return { + ...optionalLinkProperties, + link_type: LinkType.Any, + } } /** @@ -154,20 +169,24 @@ export const resolveMigrationRTImageNode = async ( * @returns Resolved link to media field. */ export const resolveMigrationLinkToMedia = ( - linkToMedia: MigrationLinkToMedia, + linkToMedia: MaybeLink, migration: Migration, ): MigrationLinkToMediaField => { const asset = migration._assets.get(linkToMedia.id.config.id)?.asset + const optionalLinkProperties = getOptionalLinkProperties(linkToMedia) if (asset) { return { + ...optionalLinkProperties, id: asset.id, link_type: LinkType.Media, - text: linkToMedia.text, } } - return { link_type: LinkType.Any } + return { + ...optionalLinkProperties, + link_type: LinkType.Any, + } } /** diff --git a/src/types/migration/Asset.ts b/src/types/migration/Asset.ts index 2eb7a1aa..de0233cb 100644 --- a/src/types/migration/Asset.ts +++ b/src/types/migration/Asset.ts @@ -73,14 +73,13 @@ export type MigrationImage = export type MigrationLinkToMedia = Pick< LinkToMediaField<"filled">, "link_type" -> & - Partial, "text">> & { - /** - * A reference to the migration asset used to resolve the link to media - * field's value. - */ - id: PrismicMigrationAsset - } +> & { + /** + * A reference to the migration asset used to resolve the link to media + * field's value. + */ + id: PrismicMigrationAsset +} /** * The minimum amount of information needed to represent a link to media field diff --git a/src/types/migration/ContentRelationship.ts b/src/types/migration/ContentRelationship.ts index 5f5edc18..83f7c6bd 100644 --- a/src/types/migration/ContentRelationship.ts +++ b/src/types/migration/ContentRelationship.ts @@ -1,5 +1,6 @@ import type { ContentRelationshipField, + EmptyContentRelationshipField, FilledContentRelationshipField, } from "../value/contentRelationship" import type { PrismicDocument } from "../value/document" @@ -27,4 +28,4 @@ export type MigrationContentRelationship< */ export type MigrationContentRelationshipField = | Pick - | ContentRelationshipField + | EmptyContentRelationshipField diff --git a/src/types/migration/Link.ts b/src/types/migration/Link.ts new file mode 100644 index 00000000..68362c43 --- /dev/null +++ b/src/types/migration/Link.ts @@ -0,0 +1,6 @@ +import type { LinkType, OptionalLinkProperties } from "../value/link" + +export type MaybeLink = + | Exclude + | (Extract & + OptionalLinkProperties) diff --git a/src/types/value/contentRelationship.ts b/src/types/value/contentRelationship.ts index 232fc443..773d6a32 100644 --- a/src/types/value/contentRelationship.ts +++ b/src/types/value/contentRelationship.ts @@ -23,7 +23,7 @@ export type ContentRelationshipField< ? EmptyContentRelationshipField : FilledContentRelationshipField -type EmptyContentRelationshipField = { +export type EmptyContentRelationshipField = { link_type: "Any" } diff --git a/src/types/value/link.ts b/src/types/value/link.ts index 0da5a35c..368c4398 100644 --- a/src/types/value/link.ts +++ b/src/types/value/link.ts @@ -31,12 +31,29 @@ export type LinkField< | Record | unknown = unknown, State extends FieldState = FieldState, -> = (State extends "empty" +> = State extends "empty" ? EmptyLinkField - : - | FilledContentRelationshipField - | FilledLinkToMediaField - | FilledLinkToWebField) & + : FilledLinkField + +/** + * A link field that is filled. + * + * @typeParam TypeEnum - Type API ID of the document. + * @typeParam LangEnum - Language API ID of the document. + * @typeParam DataInterface - Data fields for the document (filled via the + * `fetchLinks` or `graphQuery` query parameter). + */ +export type FilledLinkField< + TypeEnum = string, + LangEnum = string, + DataInterface extends + | Record + | unknown = unknown, +> = ( + | FilledContentRelationshipField + | FilledLinkToMediaField + | FilledLinkToWebField +) & OptionalLinkProperties /** @@ -45,7 +62,6 @@ export type LinkField< * @typeParam _Unused - THIS PARAMETER IS NOT USED. If you are passing a type, * **please remove it**. */ -// This type needs `OptionalLinkProperties` because this type may be used on its own. export type EmptyLinkField< _Unused extends (typeof LinkType)[keyof typeof LinkType] = typeof LinkType.Any, @@ -56,7 +72,6 @@ export type EmptyLinkField< /** * A link field pointing to a relative or absolute URL. */ -// This type needs `OptionalLinkProperties` because this type may be used on its own. export type FilledLinkToWebField = { link_type: "Web" url: string @@ -70,6 +85,8 @@ export type FilledLinkToWebField = { * * @internal */ +// Remember to update the `getOptionalLinkProperties()` function when updating +// this type. The function should check for every property. export type OptionalLinkProperties = { text?: string } diff --git a/src/types/value/richText.ts b/src/types/value/richText.ts index 3156c4aa..f6e2a854 100644 --- a/src/types/value/richText.ts +++ b/src/types/value/richText.ts @@ -1,9 +1,7 @@ import type { FieldState } from "./types" -import type { FilledContentRelationshipField } from "./contentRelationship" import type { EmbedField } from "./embed" -import type { FilledLinkToWebField } from "./link" -import type { FilledLinkToMediaField } from "./linkToMedia" +import type { FilledLinkField } from "./link" /** * Types enum for RichTextNodes @@ -176,10 +174,7 @@ export type RTImageNode = { zoom: number background: string } - linkTo?: - | FilledContentRelationshipField - | FilledLinkToWebField - | FilledLinkToMediaField + linkTo?: FilledLinkField } /** @@ -199,10 +194,7 @@ export type RTEmbedNode = { */ export interface RTLinkNode extends RTSpanNodeBase { type: typeof RichTextNodeType.hyperlink - data: - | FilledContentRelationshipField - | FilledLinkToWebField - | FilledLinkToMediaField + data: FilledLinkField } // Serialization related nodes From 8217ae665e193459c77c735f6e046c1323176414 Mon Sep 17 00:00:00 2001 From: Angelo Ashmore Date: Tue, 17 Dec 2024 12:26:55 -1000 Subject: [PATCH 07/12] docs: add `MaybeLink` description --- src/types/migration/Link.ts | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/src/types/migration/Link.ts b/src/types/migration/Link.ts index 68362c43..81f978c4 100644 --- a/src/types/migration/Link.ts +++ b/src/types/migration/Link.ts @@ -1,5 +1,18 @@ import type { LinkType, OptionalLinkProperties } from "../value/link" +/** + * Adds `OptionalLinkProperties` to any type that looks like a link object (one + * that includes a valid `link_type` property). + * + * @example + * + * ```ts + * type Example = MaybeLink + * // PrismicDocument | (LinkField & OptionalLinkProperties) + * ``` + * + * @typeParam T - The type to augment. + */ export type MaybeLink = | Exclude | (Extract & From c0f32ef48f2bb5e13126a407abcf931f53c89c50 Mon Sep 17 00:00:00 2001 From: Angelo Ashmore Date: Tue, 17 Dec 2024 12:32:22 -1000 Subject: [PATCH 08/12] test: update test --- src/lib/getOptionalLinkProperties.ts | 4 + ...ate-patch-contentRelationship.test.ts.snap | 182 ++++++++++++++++++ ...-migrate-patch-contentRelationship.test.ts | 27 ++- 3 files changed, 209 insertions(+), 4 deletions(-) diff --git a/src/lib/getOptionalLinkProperties.ts b/src/lib/getOptionalLinkProperties.ts index de83ea2a..7b1c5e18 100644 --- a/src/lib/getOptionalLinkProperties.ts +++ b/src/lib/getOptionalLinkProperties.ts @@ -15,6 +15,10 @@ export const getOptionalLinkProperties = ( ): OptionalLinkProperties => { const res: OptionalLinkProperties = {} + if ("variant" in input) { + res.variant = input.variant + } + if ("text" in input) { res.text = input.text } diff --git a/test/__snapshots__/writeClient-migrate-patch-contentRelationship.test.ts.snap b/test/__snapshots__/writeClient-migrate-patch-contentRelationship.test.ts.snap index 3b2282d6..a0e6852e 100644 --- a/test/__snapshots__/writeClient-migrate-patch-contentRelationship.test.ts.snap +++ b/test/__snapshots__/writeClient-migrate-patch-contentRelationship.test.ts.snap @@ -266,6 +266,97 @@ exports[`patches content relationship fields (from Prismic) > withText > static } `; +exports[`patches content relationship fields (from Prismic) > withVariant > group 1`] = ` +{ + "group": [ + { + "field": { + "id": "other.id-fromPrismic", + "link_type": "Document", + "variant": "Secondary", + }, + }, + ], +} +`; + +exports[`patches content relationship fields (from Prismic) > withVariant > shared slice 1`] = ` +{ + "slices": [ + { + "id": "9b9dd0f2c3f", + "items": [ + { + "field": { + "id": "other.id-fromPrismic", + "link_type": "Document", + "variant": "Secondary", + }, + }, + ], + "primary": { + "field": { + "id": "other.id-fromPrismic", + "link_type": "Document", + "variant": "Secondary", + }, + "group": [ + { + "field": { + "id": "other.id-fromPrismic", + "link_type": "Document", + "variant": "Secondary", + }, + }, + ], + }, + "slice_label": null, + "slice_type": "nunc", + "variation": "ullamcorper", + "version": "8bfc905", + }, + ], +} +`; + +exports[`patches content relationship fields (from Prismic) > withVariant > slice 1`] = ` +{ + "slices": [ + { + "id": "5306297c5ed", + "items": [ + { + "field": { + "id": "other.id-fromPrismic", + "link_type": "Document", + "variant": "Secondary", + }, + }, + ], + "primary": { + "field": { + "id": "other.id-fromPrismic", + "link_type": "Document", + "variant": "Secondary", + }, + }, + "slice_label": "Vel", + "slice_type": "hac_habitasse", + }, + ], +} +`; + +exports[`patches content relationship fields (from Prismic) > withVariant > static zone 1`] = ` +{ + "field": { + "id": "other.id-fromPrismic", + "link_type": "Document", + "variant": "Secondary", + }, +} +`; + exports[`patches content relationship fields > existing > group 1`] = ` { "group": [ @@ -525,6 +616,97 @@ exports[`patches content relationship fields > existingLongFormWithText > static } `; +exports[`patches content relationship fields > existingLongFormWithVariant > group 1`] = ` +{ + "group": [ + { + "field": { + "id": "other.id-existing", + "link_type": "Document", + "variant": "Secondary", + }, + }, + ], +} +`; + +exports[`patches content relationship fields > existingLongFormWithVariant > shared slice 1`] = ` +{ + "slices": [ + { + "id": "9b9dd0f2c3f", + "items": [ + { + "field": { + "id": "other.id-existing", + "link_type": "Document", + "variant": "Secondary", + }, + }, + ], + "primary": { + "field": { + "id": "other.id-existing", + "link_type": "Document", + "variant": "Secondary", + }, + "group": [ + { + "field": { + "id": "other.id-existing", + "link_type": "Document", + "variant": "Secondary", + }, + }, + ], + }, + "slice_label": null, + "slice_type": "nunc", + "variation": "ullamcorper", + "version": "8bfc905", + }, + ], +} +`; + +exports[`patches content relationship fields > existingLongFormWithVariant > slice 1`] = ` +{ + "slices": [ + { + "id": "5306297c5ed", + "items": [ + { + "field": { + "id": "other.id-existing", + "link_type": "Document", + "variant": "Secondary", + }, + }, + ], + "primary": { + "field": { + "id": "other.id-existing", + "link_type": "Document", + "variant": "Secondary", + }, + }, + "slice_label": "Vel", + "slice_type": "hac_habitasse", + }, + ], +} +`; + +exports[`patches content relationship fields > existingLongFormWithVariant > static zone 1`] = ` +{ + "field": { + "id": "other.id-existing", + "link_type": "Document", + "variant": "Secondary", + }, +} +`; + exports[`patches content relationship fields > lazyExisting > group 1`] = ` { "group": [ diff --git a/test/writeClient-migrate-patch-contentRelationship.test.ts b/test/writeClient-migrate-patch-contentRelationship.test.ts index 3036480c..46334e83 100644 --- a/test/writeClient-migrate-patch-contentRelationship.test.ts +++ b/test/writeClient-migrate-patch-contentRelationship.test.ts @@ -25,6 +25,13 @@ testMigrationFieldPatching< text: "foo", } }, + existingLongFormWithVariant: ({ existingDocuments }) => { + return { + link_type: LinkType.Document, + id: existingDocuments[0], + variant: "Secondary", + } + }, otherCreate: ({ otherCreateDocument }) => otherCreateDocument, lazyExisting: ({ existingDocuments }) => { return () => existingDocuments[0] @@ -86,11 +93,23 @@ testMigrationFieldPatching( contentRelationship.id = otherFromPrismicDocument.originalPrismicDocument!.id - // TODO: Remove when link text PR is merged - // @ts-expect-error - Future-proofing for link text - contentRelationship.text = "foo" + return { + ...contentRelationship, + text: "foo", + } + }, + withVariant: ({ ctx, otherFromPrismicDocument }) => { + const contentRelationship = ctx.mock.value.link({ + type: LinkType.Document, + }) + // `migrationDocuments` contains documents from "another repository" + contentRelationship.id = + otherFromPrismicDocument.originalPrismicDocument!.id - return contentRelationship + return { + ...contentRelationship, + variant: "Secondary", + } }, broken: () => { return { From 6f89eeeeaf3b0f01e6c1ed2d0e3f5c60627445a4 Mon Sep 17 00:00:00 2001 From: Angelo Ashmore Date: Tue, 17 Dec 2024 12:52:11 -1000 Subject: [PATCH 09/12] fix: add optional link fields to link to media migration fields --- src/lib/isMigrationValue.ts | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/src/lib/isMigrationValue.ts b/src/lib/isMigrationValue.ts index 72613978..a0628db8 100644 --- a/src/lib/isMigrationValue.ts +++ b/src/lib/isMigrationValue.ts @@ -85,6 +85,10 @@ export const image = (value: UnknownValue): value is MigrationImage => { /** * Checks if a value is a migration link to media field. * + * @remarks + * `OptionalLinkProperties` is included because `MigrationContentRelationship` + * may be a link field, not strictly a content relationship field. + * * @param value - Value to check. * * @returns `true` if `value` is a migration link to media field, `false` @@ -95,7 +99,7 @@ export const image = (value: UnknownValue): value is MigrationImage => { */ export const linkToMedia = ( value: UnknownValue, -): value is MigrationLinkToMedia => { +): value is MigrationLinkToMedia & OptionalLinkProperties => { return ( typeof value === "object" && value !== null && From 27839cf3149215e94b8fafa099910a9492ba2488 Mon Sep 17 00:00:00 2001 From: Angelo Ashmore Date: Tue, 17 Dec 2024 12:54:24 -1000 Subject: [PATCH 10/12] fix(migration): add missing `text` property on link to media --- src/types/migration/Asset.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/types/migration/Asset.ts b/src/types/migration/Asset.ts index de0233cb..401a07f8 100644 --- a/src/types/migration/Asset.ts +++ b/src/types/migration/Asset.ts @@ -72,7 +72,7 @@ export type MigrationImage = */ export type MigrationLinkToMedia = Pick< LinkToMediaField<"filled">, - "link_type" + "link_type" | "text" > & { /** * A reference to the migration asset used to resolve the link to media From da023890fab663d41446bdeef640a5e1065d39c6 Mon Sep 17 00:00:00 2001 From: Angelo Ashmore Date: Tue, 17 Dec 2024 12:58:00 -1000 Subject: [PATCH 11/12] Revert "fix: add optional link fields to link to media migration fields" This reverts commit 6f89eeeeaf3b0f01e6c1ed2d0e3f5c60627445a4. --- src/lib/isMigrationValue.ts | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/src/lib/isMigrationValue.ts b/src/lib/isMigrationValue.ts index a0628db8..72613978 100644 --- a/src/lib/isMigrationValue.ts +++ b/src/lib/isMigrationValue.ts @@ -85,10 +85,6 @@ export const image = (value: UnknownValue): value is MigrationImage => { /** * Checks if a value is a migration link to media field. * - * @remarks - * `OptionalLinkProperties` is included because `MigrationContentRelationship` - * may be a link field, not strictly a content relationship field. - * * @param value - Value to check. * * @returns `true` if `value` is a migration link to media field, `false` @@ -99,7 +95,7 @@ export const image = (value: UnknownValue): value is MigrationImage => { */ export const linkToMedia = ( value: UnknownValue, -): value is MigrationLinkToMedia & OptionalLinkProperties => { +): value is MigrationLinkToMedia => { return ( typeof value === "object" && value !== null && From d5cdf37e585bee12b57b8d7471f1d5152217eaf8 Mon Sep 17 00:00:00 2001 From: Angelo Ashmore Date: Tue, 17 Dec 2024 12:58:07 -1000 Subject: [PATCH 12/12] Revert "fix(migration): add missing `text` property on link to media" This reverts commit 27839cf3149215e94b8fafa099910a9492ba2488. --- src/types/migration/Asset.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/types/migration/Asset.ts b/src/types/migration/Asset.ts index 401a07f8..de0233cb 100644 --- a/src/types/migration/Asset.ts +++ b/src/types/migration/Asset.ts @@ -72,7 +72,7 @@ export type MigrationImage = */ export type MigrationLinkToMedia = Pick< LinkToMediaField<"filled">, - "link_type" | "text" + "link_type" > & { /** * A reference to the migration asset used to resolve the link to media