-
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
454 additions
and
8 deletions.
There are no files selected for viewing
259 changes: 259 additions & 0 deletions
259
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,259 @@ | ||
import { | ||
type CrossDatasetType, | ||
type SchemaType, | ||
type SearchConfiguration, | ||
type SlugSchemaType, | ||
} 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}, | ||
} | ||
|
||
// Any object type whose fields should not be considered for custom weighting. | ||
// | ||
// Search may still match on their fields, but will not traverse their schema to find custom | ||
// weights. | ||
// | ||
// Some types, such as `slug`, may instead determine weights using a specialised implementation. | ||
const ignoredBuiltInObjectTypes = ['reference', 'crossDatasetReference', 'slug'] | ||
|
||
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 isSlugField = (schemaType: SchemaType | undefined): schemaType is SlugSchemaType => { | ||
const typeChain = getTypeChain(schemaType) | ||
return typeChain.some(({jsonType, name}) => jsonType === 'object' && name === 'slug') | ||
} | ||
|
||
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 getFullyQualifiedPath(schemaType: SchemaType, path: string): string { | ||
// Slug field weights should be applied to the object's `current` field. | ||
if (isSlugField(schemaType)) { | ||
return [path, 'current'].join('.') | ||
} | ||
|
||
return path | ||
} | ||
|
||
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}] | ||
} | ||
|
||
if (isSlugField(type)) { | ||
const weight = getWeight(type, path) | ||
if (typeof weight !== 'number') return [] | ||
return [{path: getFullyQualifiedPath(type, path), weight}] | ||
} | ||
|
||
const results: SearchWeightEntry[] = [] | ||
|
||
const objectTypes = typeChain.filter( | ||
(t): t is Extract<SchemaType, {jsonType: 'object'}> => | ||
t.jsonType === 'object' && | ||
!!t.fields?.length && | ||
!ignoredBuiltInObjectTypes.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, (type, path) => { | ||
const nested = nestedWeightsBySelectionPath[getFullyQualifiedPath(type, 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, {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.