Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add functionality to convert custom types #1149

Open
wants to merge 2 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
29 changes: 29 additions & 0 deletions src/type/registry/base.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
/** A base registry class */
export class BaseRegistry<Fn extends Function> {
private readonly map = new Map<string, Fn>;

/** Returns the entries in this registry */
Entries = () => {
return new Map(this.map)
}
/** Clears all user defined string formats */
Clear = () => {
return this.map.clear()
}
/** Deletes a registered format */
Delete = (format: string) => {
return this.map.delete(format)
}
/** Returns true if the user defined string format exists */
Has = (format: string) => {
return this.map.has(format)
}
/** Sets a validation function for a user defined string format */
Set = (format: string, func: Fn) => {
this.map.set(format, func)
}
/** Gets a validation function for a user defined string format */
Get = (format: string) => {
return this.map.get(format)
}
}
39 changes: 39 additions & 0 deletions src/type/registry/convert.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
/*--------------------------------------------------------------------------

@sinclair/typebox/type

The MIT License (MIT)

Copyright (c) 2017-2025 Haydn Paterson (sinclair) <[email protected]>

Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.

---------------------------------------------------------------------------*/
import { BaseRegistry } from './base'

export type ConvertRegistryFunction<TSchema> = (schema: TSchema, value: unknown, data?: any) => unknown

const ConvertRegistry = new BaseRegistry<ConvertRegistryFunction<any>>()

export const Entries = ConvertRegistry.Entries
export const Clear = ConvertRegistry.Clear
export const Delete = ConvertRegistry.Delete
export const Has = ConvertRegistry.Has
export const Set = ConvertRegistry.Set
export const Get = ConvertRegistry.Get
36 changes: 10 additions & 26 deletions src/type/registry/format.ts
Original file line number Diff line number Diff line change
Expand Up @@ -25,31 +25,15 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.

---------------------------------------------------------------------------*/
import { BaseRegistry } from './base'

export type FormatRegistryValidationFunction = (value: string) => boolean
/** A registry for user defined string formats */
const map = new Map<string, FormatRegistryValidationFunction>()
/** Returns the entries in this registry */
export function Entries() {
return new Map(map)
}
/** Clears all user defined string formats */
export function Clear() {
return map.clear()
}
/** Deletes a registered format */
export function Delete(format: string) {
return map.delete(format)
}
/** Returns true if the user defined string format exists */
export function Has(format: string) {
return map.has(format)
}
/** Sets a validation function for a user defined string format */
export function Set(format: string, func: FormatRegistryValidationFunction) {
map.set(format, func)
}
/** Gets a validation function for a user defined string format */
export function Get(format: string) {
return map.get(format)
}

const FormatRegistry = new BaseRegistry<FormatRegistryValidationFunction>()

export const Entries = FormatRegistry.Entries
export const Clear = FormatRegistry.Clear
export const Delete = FormatRegistry.Delete
export const Has = FormatRegistry.Has
export const Set = FormatRegistry.Set
export const Get = FormatRegistry.Get
1 change: 1 addition & 0 deletions src/type/registry/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -26,5 +26,6 @@ THE SOFTWARE.

---------------------------------------------------------------------------*/

export * as ConvertRegistry from './convert'
export * as FormatRegistry from './format'
export * as TypeRegistry from './type'
37 changes: 10 additions & 27 deletions src/type/registry/type.ts
Original file line number Diff line number Diff line change
Expand Up @@ -25,32 +25,15 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.

---------------------------------------------------------------------------*/
import { BaseRegistry } from './base'

export type TypeRegistryValidationFunction<TSchema> = (schema: TSchema, value: unknown) => boolean
/** A registry for user defined types */

const map = new Map<string, TypeRegistryValidationFunction<any>>()
/** Returns the entries in this registry */
export function Entries() {
return new Map(map)
}
/** Clears all user defined types */
export function Clear() {
return map.clear()
}
/** Deletes a registered type */
export function Delete(kind: string) {
return map.delete(kind)
}
/** Returns true if this registry contains this kind */
export function Has(kind: string) {
return map.has(kind)
}
/** Sets a validation function for a user defined type */
export function Set<TSchema = unknown>(kind: string, func: TypeRegistryValidationFunction<TSchema>) {
map.set(kind, func)
}
/** Gets a custom validation function for a user defined type */
export function Get(kind: string) {
return map.get(kind)
}

const TypeRegistry = new BaseRegistry<TypeRegistryValidationFunction<any>>()

export const Entries = TypeRegistry.Entries
export const Clear = TypeRegistry.Clear
export const Delete = TypeRegistry.Delete
export const Has = TypeRegistry.Has
export const Set = TypeRegistry.Set
export const Get = TypeRegistry.Get
108 changes: 58 additions & 50 deletions src/value/convert/convert.ts
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@ import { Check } from '../check/index'
import { Deref, Pushref } from '../deref/index'
import { Kind } from '../../type/symbols/index'

import { ConvertRegistry } from '../../type/registry/index'
import type { TSchema } from '../../type/schema/index'
import type { TArray } from '../../type/array/index'
import type { TBigInt } from '../../type/bigint/index'
Expand Down Expand Up @@ -165,146 +166,153 @@ function Default(value: unknown): unknown {
// ------------------------------------------------------------------
// Convert
// ------------------------------------------------------------------
function FromArray(schema: TArray, references: TSchema[], value: any): any {
function FromArray(schema: TArray, references: TSchema[], value: any, data?: Map<string, any>): any {
const elements = IsArray(value) ? value : [value]
return elements.map((element) => Visit(schema.items, references, element))
return elements.map((element) => Visit(schema.items, references, element, data))
}
function FromBigInt(schema: TBigInt, references: TSchema[], value: any): unknown {
function FromBigInt(schema: TBigInt, references: TSchema[], value: any, data?: Map<string, any>): unknown {
return TryConvertBigInt(value)
}
function FromBoolean(schema: TBoolean, references: TSchema[], value: any): unknown {
function FromBoolean(schema: TBoolean, references: TSchema[], value: any, data?: Map<string, any>): unknown {
return TryConvertBoolean(value)
}
function FromDate(schema: TDate, references: TSchema[], value: any): unknown {
function FromDate(schema: TDate, references: TSchema[], value: any, data?: Map<string, any>): unknown {
return TryConvertDate(value)
}
function FromImport(schema: TImport, references: TSchema[], value: unknown): unknown {
function FromImport(schema: TImport, references: TSchema[], value: unknown, data?: Map<string, any>): unknown {
const definitions = globalThis.Object.values(schema.$defs) as TSchema[]
const target = schema.$defs[schema.$ref] as TSchema
return Visit(target, [...references, ...definitions], value)
return Visit(target, [...references, ...definitions], value, data)
}
function FromInteger(schema: TInteger, references: TSchema[], value: any): unknown {
function FromInteger(schema: TInteger, references: TSchema[], value: any, data?: Map<string, any>): unknown {
return TryConvertInteger(value)
}
function FromIntersect(schema: TIntersect, references: TSchema[], value: any): unknown {
return schema.allOf.reduce((value, schema) => Visit(schema, references, value), value)
function FromIntersect(schema: TIntersect, references: TSchema[], value: any, data?: Map<string, any>): unknown {
return schema.allOf.reduce((value, schema) => Visit(schema, references, value, data), value)
}
function FromLiteral(schema: TLiteral, references: TSchema[], value: any): unknown {
function FromLiteral(schema: TLiteral, references: TSchema[], value: any, data?: Map<string, any>): unknown {
return TryConvertLiteral(schema, value)
}
function FromNull(schema: TNull, references: TSchema[], value: any): unknown {
function FromNull(schema: TNull, references: TSchema[], value: any, data?: Map<string, any>): unknown {
return TryConvertNull(value)
}
function FromNumber(schema: TNumber, references: TSchema[], value: any): unknown {
function FromNumber(schema: TNumber, references: TSchema[], value: any, data?: Map<string, any>): unknown {
return TryConvertNumber(value)
}
// prettier-ignore
function FromObject(schema: TObject, references: TSchema[], value: any): unknown {
function FromObject(schema: TObject, references: TSchema[], value: any, data?: Map<string, any>): unknown {
if(!IsObject(value)) return value
for(const propertyKey of Object.getOwnPropertyNames(schema.properties)) {
if(!HasPropertyKey(value, propertyKey)) continue
value[propertyKey] = Visit(schema.properties[propertyKey], references, value[propertyKey])
value[propertyKey] = Visit(schema.properties[propertyKey], references, value[propertyKey], data)
}
return value
}
function FromRecord(schema: TRecord, references: TSchema[], value: any): unknown {
function FromRecord(schema: TRecord, references: TSchema[], value: any, data?: Map<string, any>): unknown {
const isConvertable = IsObject(value)
if (!isConvertable) return value
const propertyKey = Object.getOwnPropertyNames(schema.patternProperties)[0]
const property = schema.patternProperties[propertyKey]
for (const [propKey, propValue] of Object.entries(value)) {
value[propKey] = Visit(property, references, propValue)
value[propKey] = Visit(property, references, propValue, data)
}
return value
}
function FromRef(schema: TRef, references: TSchema[], value: any): unknown {
return Visit(Deref(schema, references), references, value)
function FromRef(schema: TRef, references: TSchema[], value: any, data?: Map<string, any>): unknown {
return Visit(Deref(schema, references), references, value, data)
}
function FromString(schema: TString, references: TSchema[], value: any): unknown {
function FromString(schema: TString, references: TSchema[], value: any, data?: Map<string, any>): unknown {
return TryConvertString(value)
}
function FromSymbol(schema: TSymbol, references: TSchema[], value: any): unknown {
function FromSymbol(schema: TSymbol, references: TSchema[], value: any, data?: Map<string, any>): unknown {
return IsString(value) || IsNumber(value) ? Symbol(value) : value
}
function FromThis(schema: TThis, references: TSchema[], value: any): unknown {
return Visit(Deref(schema, references), references, value)
function FromThis(schema: TThis, references: TSchema[], value: any, data?: Map<string, any>): unknown {
return Visit(Deref(schema, references), references, value, data)
}
// prettier-ignore
function FromTuple(schema: TTuple, references: TSchema[], value: any): unknown {
function FromTuple(schema: TTuple, references: TSchema[], value: any, data?: Map<string, any>): unknown {
const isConvertable = IsArray(value) && !IsUndefined(schema.items)
if(!isConvertable) return value
return value.map((value, index) => {
return (index < schema.items!.length)
? Visit(schema.items![index], references, value)
? Visit(schema.items![index], references, value, data)
: value
})
}
function FromUndefined(schema: TUndefined, references: TSchema[], value: any): unknown {
function FromUndefined(schema: TUndefined, references: TSchema[], value: any, data?: Map<string, any>): unknown {
return TryConvertUndefined(value)
}
function FromUnion(schema: TUnion, references: TSchema[], value: any): unknown {
function FromUnion(schema: TUnion, references: TSchema[], value: any, data?: Map<string, any>): unknown {
for (const subschema of schema.anyOf) {
const converted = Visit(subschema, references, Clone(value))
const converted = Visit(subschema, references, Clone(value), data)
if (!Check(subschema, references, converted)) continue
return converted
}
return value
}
function Visit(schema: TSchema, references: TSchema[], value: any): unknown {
function FromKind(schema: TSchema, references: TSchema[], value: unknown, data?: Map<string, any>): unknown {
if (!ConvertRegistry.Has(schema[Kind])) return Default(value)
const func = ConvertRegistry.Get(schema[Kind])!
return func(schema, value, data && data.get(schema[Kind]))
}
function Visit(schema: TSchema, references: TSchema[], value: any, data?: Map<string, any>): unknown {
const references_ = Pushref(schema, references)
const schema_ = schema as any
switch (schema[Kind]) {
case 'Array':
return FromArray(schema_, references_, value)
return FromArray(schema_, references_, value, data)
case 'BigInt':
return FromBigInt(schema_, references_, value)
return FromBigInt(schema_, references_, value, data)
case 'Boolean':
return FromBoolean(schema_, references_, value)
return FromBoolean(schema_, references_, value, data)
case 'Date':
return FromDate(schema_, references_, value)
return FromDate(schema_, references_, value, data)
case 'Import':
return FromImport(schema_, references_, value)
return FromImport(schema_, references_, value, data)
case 'Integer':
return FromInteger(schema_, references_, value)
return FromInteger(schema_, references_, value, data)
case 'Intersect':
return FromIntersect(schema_, references_, value)
return FromIntersect(schema_, references_, value, data)
case 'Literal':
return FromLiteral(schema_, references_, value)
return FromLiteral(schema_, references_, value, data)
case 'Null':
return FromNull(schema_, references_, value)
return FromNull(schema_, references_, value, data)
case 'Number':
return FromNumber(schema_, references_, value)
return FromNumber(schema_, references_, value, data)
case 'Object':
return FromObject(schema_, references_, value)
return FromObject(schema_, references_, value, data)
case 'Record':
return FromRecord(schema_, references_, value)
return FromRecord(schema_, references_, value, data)
case 'Ref':
return FromRef(schema_, references_, value)
return FromRef(schema_, references_, value, data)
case 'String':
return FromString(schema_, references_, value)
return FromString(schema_, references_, value, data)
case 'Symbol':
return FromSymbol(schema_, references_, value)
return FromSymbol(schema_, references_, value, data)
case 'This':
return FromThis(schema_, references_, value)
return FromThis(schema_, references_, value, data)
case 'Tuple':
return FromTuple(schema_, references_, value)
return FromTuple(schema_, references_, value, data)
case 'Undefined':
return FromUndefined(schema_, references_, value)
return FromUndefined(schema_, references_, value, data)
case 'Union':
return FromUnion(schema_, references_, value)
return FromUnion(schema_, references_, value, data)
default:
return Default(value)
return FromKind(schema, references_, value, data)
}
}
// ------------------------------------------------------------------
// Convert
// ------------------------------------------------------------------
/** `[Mutable]` Converts any type mismatched values to their target type if a reasonable conversion is possible. */
export function Convert(schema: TSchema, references: TSchema[], value: unknown, data?: Map<string, any>): unknown
/** `[Mutable]` Converts any type mismatched values to their target type if a reasonable conversion is possible. */
export function Convert(schema: TSchema, references: TSchema[], value: unknown): unknown
/** `[Mutable]` Converts any type mismatched values to their target type if a reasonable conversion is possible. */
export function Convert(schema: TSchema, value: unknown): unknown
/** `[Mutable]` Converts any type mismatched values to their target type if a reasonable conversion is possible. */
// prettier-ignore
export function Convert(...args: any[]) {
return args.length === 3 ? Visit(args[0], args[1], args[2]) : Visit(args[0], [], args[1])
return args.length >= 3 ? Visit(args[0], args[1], args[2], args[3]) : Visit(args[0], [], args[1])
}
Loading