@sanity/migrate
Version:
Tooling for running data migrations on Sanity.io projects
269 lines (254 loc) • 8.37 kB
text/typescript
import arrify from 'arrify'
import {type AnyArray, type ArrayElement, type NormalizeReadOnlyArray} from '../typeUtils.js'
import {
type DecOp,
type DiffMatchPatchOp,
type IncOp,
type IndexedSegment,
type InsertOp,
type KeyedSegment,
type RelativePosition,
type ReplaceOp,
type SetIfMissingOp,
type SetOp,
type TruncateOp,
type UnsetOp,
} from './types.js'
/**
* Creates a `set` operation with the provided value.
* @param value - The value to set.
* @returns A `set` operation.
* {@link https://www.sanity.io/docs/http-patches#6TPENSW3}
*
* @example
* ```ts
* const setFoo = set('foo')
* const setEmptyArray = set([])
* ```
*/
export const set = <const T>(value: T): SetOp<T> => ({type: 'set', value})
/**
* Creates a `setIfMissing` operation with the provided value.
* @param value - The value to set if missing.
* @returns A `setIfMissing` operation.
* {@link https://www.sanity.io/docs/http-patches#A80781bT}
* @example
* ```ts
* const setFooIfMissing = setIfMissing('foo')
* const setEmptyArrayIfMissing = setIfMissing([])
* ```
*/
export const setIfMissing = <const T>(value: T): SetIfMissingOp<T> => ({
type: 'setIfMissing',
value,
})
/**
* Creates an `unset` operation.
* @returns An `unset` operation.
* {@link https://www.sanity.io/docs/http-patches#xRtBjp8o}
*
* @example
* ```ts
* const unsetAnyValue = unset()
* ```
*/
export const unset = (): UnsetOp => ({type: 'unset'})
/**
* Creates an `inc` (increment) operation with the provided amount.
* @param amount - The amount to increment by.
* @returns An incrementation operation for numeric values
* {@link https://www.sanity.io/docs/http-patches#vIT8WWQo}
*
* @example
* ```ts
* const incBy1 = inc()
* const incBy5 = inc(5)
* ```
*/
export const inc = <const N extends number = 1>(amount: N = 1 as N): IncOp<N> => ({
amount,
type: 'inc',
})
/**
* Creates a `dec` (decrement) operation with the provided amount.
* @param amount - The amount to decrement by.
* @returns A `dec` operation.
* {@link https://www.sanity.io/docs/http-patches#vIT8WWQo}
*
* @example
* ```ts
* const decBy1 = dec()
* const decBy10 = dec(10)
* ```
*/
export const dec = <const N extends number = 1>(amount: N = 1 as N): DecOp<N> => ({
amount,
type: 'dec',
})
/**
* Creates a `diffMatchPatch` operation with the provided value.
* @param value - The value for the diff match patch operation.
* @returns A `diffMatchPatch` operation.
* {@link https://www.sanity.io/docs/http-patches#aTbJhlAJ}
* @public
*/
export const diffMatchPatch = (value: string): DiffMatchPatchOp => ({
type: 'diffMatchPatch',
value,
})
/**
* Creates an `insert` operation with the provided items, position, and reference item.
* @param items - The items to insert.
* @param position - The position to insert at.
* @param indexOrReferenceItem - The index or reference item to insert before or after.
* @returns An `insert` operation for adding values to arrays
* {@link https://www.sanity.io/docs/http-patches#febxf6Fk}
*
* @example
* ```ts
* const prependFoo = insert(['foo'], 'before', 0)
* const appendFooAndBar = insert(['foo', 'bar'], 'after', someArray.length -1)
* const insertObjAfterXYZ = insert({name: 'foo'}, 'after', {_key: 'xyz'}])
* ```
*/
export function insert<
const Items extends AnyArray<unknown>,
const Pos extends RelativePosition,
const ReferenceItem extends IndexedSegment | KeyedSegment,
>(
items: ArrayElement<Items> | Items,
position: Pos,
indexOrReferenceItem: ReferenceItem,
): InsertOp<NormalizeReadOnlyArray<Items>, Pos, ReferenceItem> {
return {
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- arrify loses type information, cast needed for tuple type preservation
items: arrify(items) as any,
position,
referenceItem: indexOrReferenceItem,
type: 'insert',
}
}
/**
* Creates an `insert` operation that appends the provided items.
* @param items - The items to append.
* @returns An `insert` operation for adding a value to the end of an array.
* {@link https://www.sanity.io/docs/http-patches#Cw4vhD88}
*
* @example
* ```ts
* const appendFoo = append('foo')
* const appendObject = append({name: 'foo'})
* const appendObjects = append([{name: 'foo'}, [{name: 'bar'}]])
* ```
*/
export function append<const Items extends AnyArray<unknown>>(items: ArrayElement<Items> | Items) {
return insert(items, 'after', -1)
}
/**
* Creates an `insert` operation that prepends the provided items.
* @param items - The items to prepend.
* @returns An `insert` operation for adding a value to the start of an array.
* {@link https://www.sanity.io/docs/http-patches#refAUsf0}
*
* @example
* ```ts
* const prependFoo = prepend('foo')
* const prependObject = prepend({name: 'foo'})
* const prependObjects = prepend([{name: 'foo'}, [{name: 'bar'}]])
* ```
*/
export function prepend<const Items extends AnyArray<unknown>>(items: ArrayElement<Items> | Items) {
return insert(items, 'before', 0)
}
/**
* Creates an `insert` operation that inserts the provided items before the provided index or reference item.
* @param items - The items to insert.
* @param indexOrReferenceItem - The index or reference item to insert before.
* @returns An `insert` operation before the provided index or reference item.
* {@link https://www.sanity.io/docs/http-patches#0SQmPlb6}
* @public
*
* @example
* ```ts
* const insertFooBeforeIndex3 = insertBefore('foo', 3)
* const insertObjectBeforeKey = insertBefore({name: 'foo'}, {_key: 'xyz'}]
* ```
*/
export function insertBefore<
const Items extends AnyArray<unknown>,
const ReferenceItem extends IndexedSegment | KeyedSegment,
>(items: ArrayElement<Items> | Items, indexOrReferenceItem: ReferenceItem) {
return insert(items, 'before', indexOrReferenceItem)
}
/**
* Creates an `insert` operation that inserts the provided items after the provided index or reference item.
* @param items - The items to insert.
* @param indexOrReferenceItem - The index or reference item to insert after.
* @returns An `insert` operation after the provided index or reference item.
* {@link https://www.sanity.io/docs/http-patches#0SQmPlb6}
*
* @example
* ```ts
* const insertFooAfterIndex3 = insertAfter('foo', 3)
* const insertObjectAfterKey = insertAfter({name: 'foo'}, {_key: 'xyz'}]
* ```
*/
export const insertAfter = <
const Items extends AnyArray<unknown>,
const ReferenceItem extends IndexedSegment | KeyedSegment,
>(
items: ArrayElement<Items> | Items,
indexOrReferenceItem: ReferenceItem,
) => {
return insert(items, 'after', indexOrReferenceItem)
}
/**
* Creates a `truncate` operation that will remove all items after `startIndex` until the end of the array or the provided `endIndex`.
* @param startIndex - The start index for the truncate operation.
* @param endIndex - The end index for the truncate operation.
* @returns A `truncate` operation.
* @remarks - This will be converted to an `unset` patch when submitted to the API
* {@link https://www.sanity.io/docs/http-patches#xRtBjp8o}
*
* @example
* ```ts
* const clearArray = truncate(0)
* const removeItems = truncate(3, 5) // Removes items at index 3, 4, and 5
* const truncate200 = truncate(200) // Removes all items after index 200
* ```
*/
export function truncate(startIndex: number, endIndex?: number): TruncateOp {
return {
startIndex,
type: 'truncate',
...(endIndex !== undefined && {endIndex}),
}
}
/**
* Creates a `replace` operation with the provided items and reference item.
* @param items - The items to replace.
* @param referenceItem - The reference item to replace.
* @returns A ReplaceOp operation.
* @remarks This will be converted to an `insert`/`replace` patch when submitted to the API
* {@link https://www.sanity.io/docs/http-patches#GnVSwcPa}
*
* @example
* ```ts
* const replaceItem3WithFoo = replace('foo', 3)
* const replaceItem3WithFooAndBar = replace(['foo', 'bar'], 3)
* const replaceObject = replace({name: 'bar'}, {_key: 'xyz'})
* ```
*/
export function replace<
Items extends unknown[],
ReferenceItem extends IndexedSegment | KeyedSegment,
>(
items: ArrayElement<Items> | Items,
referenceItem: ReferenceItem,
): ReplaceOp<Items, ReferenceItem> {
return {
items: arrify(items) as Items,
referenceItem,
type: 'replace',
}
}