-
Notifications
You must be signed in to change notification settings - Fork 435
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
6 changed files
with
415 additions
and
3 deletions.
There are no files selected for viewing
223 changes: 223 additions & 0 deletions
223
packages/sanity/src/core/search/common/deriveSearchWeightsFromType2024.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,223 @@ | ||
import {type CrossDatasetType, type SchemaType, type SearchConfiguration} from '@sanity/types' | ||
import {toString as pathToString} from '@sanity/util/paths' | ||
|
||
import {isRecord} from '../../util' | ||
import {type DeriveSearchWeightsFromTypeOptions} from './deriveSearchWeightsFromType' | ||
import {type SearchSpec} from './types' | ||
|
||
interface SearchWeightEntry { | ||
path: string | ||
weight: number | ||
type?: 'string' | 'pt' | ||
} | ||
|
||
const CACHE = new WeakMap<SchemaType | CrossDatasetType, SearchSpec>() | ||
const PREVIEW_FIELD_WEIGHT_MAP = { | ||
title: 10, | ||
subtitle: 5, | ||
description: 1.5, | ||
} | ||
const BASE_WEIGHTS: Record<string, Omit<SearchWeightEntry, 'path'>> = { | ||
_id: {weight: 1}, | ||
_type: {weight: 1}, | ||
} | ||
const builtInObjectTypes = ['reference', 'crossDatasetReference'] | ||
|
||
const getTypeChain = (type: SchemaType | undefined): SchemaType[] => | ||
type ? [type, ...getTypeChain(type.type)] : [] | ||
|
||
const isPtField = (type: SchemaType | undefined) => | ||
type?.jsonType === 'array' && | ||
type.of.some((arrType) => getTypeChain(arrType).some(({name}) => name === 'block')) | ||
|
||
const isStringField = (schemaType: SchemaType | undefined): boolean => | ||
schemaType ? schemaType?.jsonType === 'string' : false | ||
|
||
const isSearchConfiguration = (options: unknown): options is SearchConfiguration => | ||
isRecord(options) && 'search' in options && isRecord(options.search) | ||
|
||
function isSchemaType(input: SchemaType | CrossDatasetType | undefined): input is SchemaType { | ||
return typeof input !== 'undefined' && 'name' in input | ||
} | ||
|
||
function getLeafWeights( | ||
schemaType: SchemaType | CrossDatasetType | undefined, | ||
maxDepth: number, | ||
getWeight: (schemaType: SchemaType, path: string) => number | null, | ||
): Record<string, SearchWeightEntry> { | ||
function traverse( | ||
type: SchemaType | undefined, | ||
path: string, | ||
depth: number, | ||
): SearchWeightEntry[] { | ||
if (!type) return [] | ||
if (depth > maxDepth) return [] | ||
|
||
const typeChain = getTypeChain(type) | ||
|
||
if (isStringField(type) || isPtField(type)) { | ||
const weight = getWeight(type, path) | ||
|
||
if (typeof weight !== 'number') return [] | ||
return [{path, weight}] | ||
} | ||
|
||
const results: SearchWeightEntry[] = [] | ||
const objectTypes = typeChain.filter( | ||
(t): t is Extract<SchemaType, {jsonType: 'object'}> => | ||
t.jsonType === 'object' && !!t.fields?.length && !builtInObjectTypes.includes(t.name), | ||
) | ||
for (const objectType of objectTypes) { | ||
// TODO: Allow override of aliased types. | ||
for (const field of objectType.fields) { | ||
const nextPath = pathToString([path, field.name].filter(Boolean)) | ||
results.push(...traverse(field.type, nextPath, depth + 1)) | ||
} | ||
} | ||
|
||
const arrayTypes = typeChain.filter( | ||
(t): t is Extract<SchemaType, {jsonType: 'array'}> => | ||
t.jsonType === 'array' && !!t.of?.length, | ||
) | ||
for (const arrayType of arrayTypes) { | ||
for (const arrayItemType of arrayType.of) { | ||
const nextPath = `${path}[]` | ||
results.push(...traverse(arrayItemType, nextPath, depth + 1)) | ||
} | ||
} | ||
|
||
return results | ||
} | ||
|
||
// Cross Dataset Reference are not part of the schema, so we should not attempt to reconcile them. | ||
if (!isSchemaType(schemaType)) { | ||
return {} | ||
} | ||
|
||
return traverse(schemaType, '', 0).reduce<Record<string, SearchWeightEntry>>( | ||
(acc, {path, weight, type}) => { | ||
acc[path] = {weight, type, path} | ||
return acc | ||
}, | ||
{}, | ||
) | ||
} | ||
|
||
const getUserSetWeight = (schemaType: SchemaType) => { | ||
const searchOptions = getTypeChain(schemaType) | ||
.map((type) => type.options) | ||
.find(isSearchConfiguration) | ||
|
||
return typeof searchOptions?.search?.weight === 'number' ? searchOptions.search.weight : null | ||
} | ||
|
||
const getHiddenWeight = (schemaType: SchemaType) => { | ||
const hidden = getTypeChain(schemaType).some((type) => type.hidden) | ||
return hidden ? 0 : null | ||
} | ||
|
||
const getDefaultWeights = (schemaType: SchemaType) => { | ||
// if there is no user set weight or a `0` weight due to be hidden, | ||
// then we can return the default weight of `1` | ||
const result = getUserSetWeight(schemaType) ?? getHiddenWeight(schemaType) | ||
return typeof result === 'number' ? null : 1 | ||
} | ||
|
||
const getPreviewWeights = ( | ||
schemaType: SchemaType | CrossDatasetType | undefined, | ||
maxDepth: number, | ||
isCrossDataset?: boolean, | ||
): Record<string, SearchWeightEntry> | null => { | ||
const select = schemaType?.preview?.select | ||
if (!select) return null | ||
|
||
const selectionKeysBySelectionPath = Object.fromEntries( | ||
Object.entries(select).map(([selectionKey, selectionPath]) => [ | ||
// replace indexed paths with `[]` | ||
// e.g. `arrayOfObjects.0.myField` becomes `arrayOfObjects[].myField` | ||
selectionPath.replace(/\.\d+/g, '[]'), | ||
selectionKey, | ||
]), | ||
) | ||
|
||
const defaultWeights = getLeafWeights(schemaType, maxDepth, getDefaultWeights) | ||
const nestedWeightsBySelectionPath = Object.fromEntries( | ||
Object.entries(defaultWeights) | ||
.map(([path, {type}]) => ({path, type})) | ||
.filter(({path}) => selectionKeysBySelectionPath[path]) | ||
.map(({path, type}) => [ | ||
path, | ||
{ | ||
type, | ||
weight: | ||
PREVIEW_FIELD_WEIGHT_MAP[ | ||
selectionKeysBySelectionPath[path] as keyof typeof PREVIEW_FIELD_WEIGHT_MAP | ||
], | ||
}, | ||
]), | ||
) | ||
|
||
if (isCrossDataset) { | ||
return Object.fromEntries( | ||
Object.entries(selectionKeysBySelectionPath).map(([path, previewFieldName]) => { | ||
return [ | ||
path, | ||
{ | ||
path, | ||
type: 'string', | ||
weight: | ||
PREVIEW_FIELD_WEIGHT_MAP[previewFieldName as keyof typeof PREVIEW_FIELD_WEIGHT_MAP], | ||
}, | ||
] | ||
}), | ||
) | ||
} | ||
|
||
return getLeafWeights(schemaType, maxDepth, (_, path) => { | ||
const nested = nestedWeightsBySelectionPath[path] | ||
return nested ? nested.weight : null | ||
}) | ||
} | ||
|
||
// export interface DeriveSearchWeightsFromTypeOptions { | ||
// schemaType: SchemaType | CrossDatasetType | ||
// maxDepth: number | ||
// isCrossDataset?: boolean | ||
// processPaths?: (paths: SearchPath[]) => SearchPath[] | ||
// } | ||
|
||
export function deriveSearchWeightsFromType2024({ | ||
schemaType, | ||
maxDepth, | ||
isCrossDataset, | ||
processPaths = (paths) => paths, | ||
}: DeriveSearchWeightsFromTypeOptions): SearchSpec { | ||
const cached = CACHE.get(schemaType) | ||
if (cached) return cached | ||
|
||
const userSetWeights = getLeafWeights(schemaType, maxDepth, getUserSetWeight) | ||
const hiddenWeights = getLeafWeights(schemaType, maxDepth, getHiddenWeight) | ||
const defaultWeights = getLeafWeights(schemaType, maxDepth, getDefaultWeights) | ||
const previewWeights = getPreviewWeights(schemaType, maxDepth, isCrossDataset) | ||
|
||
const weights: Record<string, Omit<SearchWeightEntry, 'path'>> = { | ||
...BASE_WEIGHTS, | ||
...defaultWeights, | ||
...hiddenWeights, | ||
...previewWeights, | ||
...userSetWeights, | ||
} | ||
|
||
const result = { | ||
typeName: isSchemaType(schemaType) ? schemaType.name : schemaType.type, | ||
paths: processPaths( | ||
Object.entries(weights).map(([path, {type, weight}]) => ({ | ||
path, | ||
weight, | ||
})), | ||
), | ||
} | ||
|
||
CACHE.set(schemaType, result) | ||
return result | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,3 +1,4 @@ | ||
export * from './deriveSearchWeightsFromType' | ||
export * from './deriveSearchWeightsFromType2024' | ||
export * from './getSearchableTypes' | ||
export * from './types' |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.