mirror of
https://github.com/Start9Labs/start-os.git
synced 2026-03-30 12:11:56 +00:00
chore: Convert from ajv to ts-matches (#1415)
This commit is contained in:
@@ -1,82 +1,92 @@
|
||||
import * as Ajv from 'ajv'
|
||||
import { applyOperation } from 'fast-json-patch'
|
||||
|
||||
import {
|
||||
Parser,
|
||||
shape,
|
||||
string,
|
||||
literal,
|
||||
boolean,
|
||||
object,
|
||||
deferred,
|
||||
dictionary,
|
||||
anyOf,
|
||||
} from 'ts-matches'
|
||||
|
||||
export type ValidVersion = 1 | 2
|
||||
|
||||
function has<Obj extends {}, K extends string>(obj: Obj, key: K): obj is (Obj & { [P in K]: unknown }) {
|
||||
function has<Obj extends {}, K extends string>(
|
||||
obj: Obj,
|
||||
key: K,
|
||||
): obj is Obj & { [P in K]: unknown } {
|
||||
return key in obj
|
||||
}
|
||||
|
||||
|
||||
const ajv = new Ajv({ jsonPointers: true, allErrors: true, nullable: true })
|
||||
const ajvWithDefaults = new Ajv({ jsonPointers: true, allErrors: true, useDefaults: true, nullable: true, removeAdditional:true })
|
||||
|
||||
const schemaV1 = {
|
||||
'type': 'object',
|
||||
'properties': {
|
||||
'name': { 'type': 'string' },
|
||||
'value': { 'type': 'string' },
|
||||
'description': { 'type': 'string', 'nullable': true, 'default': null },
|
||||
'copyable': { 'type': 'boolean', 'default': false },
|
||||
'qr': { 'type': 'boolean', 'default': false },
|
||||
const matchPropertiesV1 = shape(
|
||||
{
|
||||
name: string,
|
||||
value: string,
|
||||
description: string,
|
||||
copyable: boolean,
|
||||
qr: boolean,
|
||||
},
|
||||
'required': ['name', 'value', 'copyable', 'qr'],
|
||||
'additionalProperties': false,
|
||||
}
|
||||
const schemaV1Compiled = ajv.compile(schemaV1)
|
||||
function isSchemaV1(properties: unknown): properties is PropertiesV1 {
|
||||
return schemaV1Compiled(properties) as any
|
||||
}
|
||||
const _schemaV1CompiledWithDefaults = ajvWithDefaults.compile(schemaV1)
|
||||
function schemaV1CompiledWithDefaults(properties: unknown): properties is PropertiesV1 {
|
||||
return _schemaV1CompiledWithDefaults(properties) as any
|
||||
}
|
||||
const schemaV2 = {
|
||||
'anyOf': [
|
||||
{
|
||||
'type': 'object',
|
||||
'properties': {
|
||||
'type': { 'type': 'string', 'const': 'string' },
|
||||
'value': { 'type': 'string' },
|
||||
'description': { 'type': 'string', 'nullable': true, 'default': null },
|
||||
'copyable': { 'type': 'boolean', 'default': false },
|
||||
'qr': { 'type': 'boolean', 'default': false },
|
||||
'masked': { 'type': 'boolean', 'default': false },
|
||||
},
|
||||
'required': ['type', 'value', 'description', 'copyable', 'qr', 'masked'],
|
||||
'additionalProperties': false,
|
||||
},
|
||||
{
|
||||
'type': 'object',
|
||||
'properties': {
|
||||
'type': { 'type': 'string', 'const': 'object' },
|
||||
'value': {
|
||||
'type': 'object',
|
||||
'patternProperties': {
|
||||
'^.*$': {
|
||||
'$ref': '#',
|
||||
},
|
||||
},
|
||||
},
|
||||
'description': { 'type': 'string', 'nullable': true, 'default': null },
|
||||
['description', 'copyable', 'qr'],
|
||||
{ description: null as null, copyable: false, qr: false } as const,
|
||||
)
|
||||
type PropertiesV1 = typeof matchPropertiesV1._TYPE
|
||||
|
||||
},
|
||||
'required': ['type', 'value', 'description'],
|
||||
'additionalProperties': false,
|
||||
},
|
||||
],
|
||||
type PackagePropertiesV2 = {
|
||||
[name: string]: PackagePropertyString | PackagePropertyObject
|
||||
}
|
||||
const schemaV2Compiled = ajv.compile(schemaV2)
|
||||
const schemaV2CompiledWithDefaults = ajvWithDefaults.compile(schemaV2)
|
||||
|
||||
export function parsePropertiesPermissive (properties: unknown, errorCallback: (err: Error) => any = console.warn): PackageProperties {
|
||||
const [matchPackagePropertiesV2, setPPV2] = deferred<PackagePropertiesV2>()
|
||||
const matchPackagePropertyString = shape(
|
||||
{
|
||||
type: literal('string'),
|
||||
description: string,
|
||||
value: string,
|
||||
copyable: boolean,
|
||||
qr: boolean,
|
||||
masked: boolean,
|
||||
},
|
||||
['description', 'copyable', 'qr', 'masked'],
|
||||
{
|
||||
description: null as null,
|
||||
copyable: false,
|
||||
qr: false,
|
||||
masked: false,
|
||||
} as const,
|
||||
)
|
||||
type PackagePropertyString = typeof matchPackagePropertyString._TYPE
|
||||
const matchPackagePropertyObject = shape(
|
||||
{
|
||||
type: literal('object'),
|
||||
value: matchPackagePropertiesV2,
|
||||
description: string.optional(),
|
||||
},
|
||||
['description'],
|
||||
{ description: null as null },
|
||||
)
|
||||
const matchPropertyV2 = anyOf(
|
||||
matchPackagePropertyString,
|
||||
matchPackagePropertyObject,
|
||||
)
|
||||
type PackagePropertyObject = typeof matchPackagePropertyObject._TYPE
|
||||
setPPV2(dictionary([string, matchPropertyV2]))
|
||||
|
||||
export function parsePropertiesPermissive(
|
||||
properties: unknown,
|
||||
errorCallback: (err: Error) => any = console.warn,
|
||||
): PackageProperties {
|
||||
if (typeof properties !== 'object' || properties === null) {
|
||||
errorCallback(new TypeError(`${properties} is not an object`))
|
||||
return {}
|
||||
}
|
||||
// @TODO still need this conditional?
|
||||
if (!has(properties, 'version') || !has(properties, 'data') || typeof properties.version !== 'number' || !properties.data) {
|
||||
if (
|
||||
!has(properties, 'version') ||
|
||||
!has(properties, 'data') ||
|
||||
typeof properties.version !== 'number' ||
|
||||
!properties.data
|
||||
) {
|
||||
return Object.entries(properties)
|
||||
.filter(([_, value]) => {
|
||||
if (typeof value === 'string') {
|
||||
@@ -99,8 +109,7 @@ export function parsePropertiesPermissive (properties: unknown, errorCallback: (
|
||||
.reduce((acc, { name, value }) => {
|
||||
acc[name] = value
|
||||
return acc
|
||||
}, { })
|
||||
|
||||
}, {})
|
||||
}
|
||||
switch (properties.version) {
|
||||
case 1:
|
||||
@@ -108,124 +117,96 @@ export function parsePropertiesPermissive (properties: unknown, errorCallback: (
|
||||
case 2:
|
||||
return parsePropertiesV2Permissive(properties.data, errorCallback)
|
||||
default:
|
||||
errorCallback(new Error(`unknown properties version ${properties.version}, attempting to parse as v2`))
|
||||
errorCallback(
|
||||
new Error(
|
||||
`unknown properties version ${properties.version}, attempting to parse as v2`,
|
||||
),
|
||||
)
|
||||
return parsePropertiesV2Permissive(properties.data, errorCallback)
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
function parsePropertiesV1Permissive (properties: unknown, errorCallback: (err: Error) => any): PackageProperties {
|
||||
function parsePropertiesV1Permissive(
|
||||
properties: unknown,
|
||||
errorCallback: (err: Error) => any,
|
||||
): PackageProperties {
|
||||
if (!Array.isArray(properties)) {
|
||||
errorCallback(new TypeError(`${properties} is not an array`))
|
||||
return {}
|
||||
}
|
||||
const parsedProperties : PackagePropertiesV2 = {};
|
||||
for(const idx in properties) {
|
||||
const cur:unknown = properties[idx]
|
||||
if(isSchemaV1(cur)) {
|
||||
parsedProperties[cur.name] = {
|
||||
type: 'string',
|
||||
value: cur.value,
|
||||
description: cur.description,
|
||||
copyable: cur.copyable,
|
||||
qr: cur.qr,
|
||||
masked: false,
|
||||
}
|
||||
}
|
||||
else if (schemaV1Compiled.errors) {
|
||||
for (let err of schemaV1Compiled.errors) {
|
||||
errorCallback(new Error(`/data/${idx}${err.dataPath}: ${err.message}`))
|
||||
if (err.dataPath) {
|
||||
applyOperation(cur, { op: 'replace', path: err.dataPath, value: undefined })
|
||||
return properties.reduce(
|
||||
(prev: PackagePropertiesV2, cur: unknown, idx: number) => {
|
||||
const result = matchPropertiesV1.enumParsed(cur)
|
||||
if ('value' in result) {
|
||||
const value = result.value
|
||||
prev[value.name] = {
|
||||
type: 'string',
|
||||
value: value.value,
|
||||
description: value.description,
|
||||
copyable: value.copyable,
|
||||
qr: value.qr,
|
||||
masked: false,
|
||||
}
|
||||
} else {
|
||||
const error = result.error
|
||||
const message = Parser.validatorErrorAsString(error)
|
||||
let dataPath = error.keys.map(x => JSON.parse(x)).join('/')
|
||||
errorCallback(new Error(`/data/${idx}: ${message}`))
|
||||
if (dataPath) {
|
||||
applyOperation(cur, {
|
||||
op: 'replace',
|
||||
path: dataPath,
|
||||
value: undefined,
|
||||
})
|
||||
}
|
||||
}
|
||||
if (!schemaV1CompiledWithDefaults(cur)) {
|
||||
for (let err of _schemaV1CompiledWithDefaults.errors) {
|
||||
errorCallback(new Error(`/data/${idx}${err.dataPath}: ${err.message}`))
|
||||
}
|
||||
continue
|
||||
}
|
||||
parsedProperties[cur.name] = {
|
||||
|
||||
type: 'string',
|
||||
value: cur.value,
|
||||
description: cur.description,
|
||||
copyable: cur.copyable,
|
||||
qr: cur.qr,
|
||||
masked: false,
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
return parsedProperties
|
||||
return prev
|
||||
},
|
||||
{},
|
||||
)
|
||||
}
|
||||
function parsePropertiesV2Permissive (properties: unknown, errorCallback: (err: Error) => any): PackageProperties {
|
||||
if (typeof properties !== 'object' || properties === null) {
|
||||
errorCallback(new TypeError(`${properties} is not an object`))
|
||||
function parsePropertiesV2Permissive(
|
||||
properties: unknown,
|
||||
errorCallback: (err: Error) => any,
|
||||
): PackageProperties {
|
||||
if (!object.test(properties)) {
|
||||
return {}
|
||||
}
|
||||
return Object.entries(properties).reduce((prev, [name, value], idx) => {
|
||||
schemaV2Compiled(value)
|
||||
if (schemaV2Compiled.errors) {
|
||||
for (let err of schemaV2Compiled.errors) {
|
||||
errorCallback(new Error(`/data/${idx}${err.dataPath}: ${err.message}`))
|
||||
if (err.dataPath) {
|
||||
applyOperation(value, { op: 'replace', path: err.dataPath, value: undefined })
|
||||
return Object.entries(properties).reduce(
|
||||
(prev: PackageProperties, [name, value], idx) => {
|
||||
const result = matchPropertyV2.enumParsed(value)
|
||||
if ('value' in result) {
|
||||
prev[name] = result.value
|
||||
} else {
|
||||
const error = result.error
|
||||
const message = Parser.validatorErrorAsString(error)
|
||||
let dataPath = error.keys.map(x => JSON.parse(x)).join('/')
|
||||
errorCallback(new Error(`/data/${idx}: ${message}`))
|
||||
if (dataPath) {
|
||||
applyOperation(properties, {
|
||||
op: 'replace',
|
||||
path: dataPath,
|
||||
value: undefined,
|
||||
})
|
||||
}
|
||||
}
|
||||
if (!schemaV2CompiledWithDefaults(value)) {
|
||||
for (let err of schemaV2CompiledWithDefaults.errors) {
|
||||
errorCallback(new Error(`/data/${idx}${err.dataPath}: ${err.message}`))
|
||||
}
|
||||
return prev
|
||||
}
|
||||
}
|
||||
prev[name] = value
|
||||
return prev
|
||||
}, { })
|
||||
}
|
||||
interface PropertiesV1 {
|
||||
name: string
|
||||
value: string
|
||||
description: string | null
|
||||
copyable: boolean
|
||||
qr: boolean
|
||||
return prev
|
||||
},
|
||||
|
||||
{},
|
||||
)
|
||||
}
|
||||
|
||||
type PackagePropertiesV1 = PropertiesV1[]
|
||||
export type PackageProperties = PackagePropertiesV2
|
||||
|
||||
export type PackagePropertiesVersioned<T extends number> = {
|
||||
version: T,
|
||||
version: T
|
||||
data: PackagePropertiesVersionedData<T>
|
||||
}
|
||||
|
||||
export type PackagePropertiesVersionedData<T extends number> =
|
||||
T extends 1 ? PackagePropertiesV1 :
|
||||
T extends 2 ? PackagePropertiesV2 :
|
||||
never
|
||||
|
||||
|
||||
interface PackagePropertiesV2 {
|
||||
[name: string]: PackagePropertyString | PackagePropertyObject
|
||||
}
|
||||
|
||||
interface PackagePropertyBase {
|
||||
type: 'string' | 'object'
|
||||
description: string | null
|
||||
}
|
||||
|
||||
interface PackagePropertyString extends PackagePropertyBase {
|
||||
type: 'string'
|
||||
value: string
|
||||
copyable: boolean
|
||||
qr: boolean
|
||||
masked: boolean
|
||||
}
|
||||
|
||||
interface PackagePropertyObject extends PackagePropertyBase {
|
||||
type: 'object'
|
||||
value: PackagePropertiesV2
|
||||
}
|
||||
|
||||
export type PackagePropertiesVersionedData<T extends number> = T extends 1
|
||||
? PackagePropertiesV1
|
||||
: T extends 2
|
||||
? PackagePropertiesV2
|
||||
: never
|
||||
|
||||
Reference in New Issue
Block a user