UNPKG

@pmouli/isy-matter-server

Version:

Service to expose an ISY device as a Matter Border router

379 lines (302 loc) 16.3 kB
import { unique } from 'moderndash'; import ts, { ModifierFlags, type ModifierSyntaxKind, type PunctuationSyntaxKind } from 'typescript'; import { types } from 'util'; type ParameterType<F> = F extends (arg0: infer P) => any ? P : never; type TokenSyntaxKind = ParameterType<(typeof ts.factory)['createToken']>; export const SyntaxKind = ts.SyntaxKind; export type SyntaxKind = ts.SyntaxKind; export class CodeFactory { protected factory: ts.NodeFactory; constructor(factory: ts.NodeFactory) { this.factory = factory; } private createIdentifier(name?: string): ts.Identifier | undefined { return name ? ts.factory.createIdentifier(name) : undefined; } createImportClause(isTypeOnly: boolean, name?: string, ...namedBindings: ts.NamedImportBindings[]): ts.ImportClause { return ts.factory.createImportClause(isTypeOnly, this.createIdentifier(name), namedBindings.length ? namedBindings[0] : undefined); } createAssertClause(elements: ts.NodeArray<ts.AssertEntry>, multiLine?: boolean): ts.AssertClause { return ts.factory.createAssertClause(elements, multiLine); } createAssertEntry(name: string, value: ts.Expression): ts.AssertEntry { return ts.factory.createAssertEntry(this.createIdentifier(name)!, value); } createImportTypeAssertionContainer(clause: ts.AssertClause, multiLine?: boolean): ts.ImportTypeAssertionContainer { return ts.factory.createImportTypeAssertionContainer(clause, multiLine); } createImportAttributes(elements: ts.NodeArray<ts.ImportAttribute>, multiLine?: boolean): ts.ImportAttributes { return ts.factory.createImportAttributes(elements, multiLine); } createImportAttribute(name: string, value: ts.Expression): ts.ImportAttribute { return ts.factory.createImportAttribute(this.createIdentifier(name)!, value); } createNamespaceImport(name: string): ts.NamespaceImport { return ts.factory.createNamespaceImport(this.createIdentifier(name)!); } createNamespaceExport(name: string): ts.NamespaceExport { return ts.factory.createNamespaceExport(this.createIdentifier(name)!); } createNamedImports(...elements: ts.ImportSpecifier[]): ts.NamedImports { return ts.factory.createNamedImports(elements); } updateImportClause(node: ts.ImportClause, isTypeOnly: boolean, name?: string, ...namedBindings: ts.NamedImportBindings[]): ts.ImportClause { return ts.factory.updateImportClause(node, isTypeOnly, this.createIdentifier(name), namedBindings.length ? namedBindings[0] : undefined); } updateAssertClause(node: ts.AssertClause, elements: ts.NodeArray<ts.AssertEntry>, multiLine?: boolean): ts.AssertClause { return ts.factory.updateAssertClause(node, elements, multiLine); } updateAssertEntry(node: ts.AssertEntry, name: string, value: ts.Expression): ts.AssertEntry { return ts.factory.updateAssertEntry(node, this.createIdentifier(name)!, value); } updateImportTypeAssertionContainer(node: ts.ImportTypeAssertionContainer, clause: ts.AssertClause, multiLine?: boolean): ts.ImportTypeAssertionContainer { return ts.factory.updateImportTypeAssertionContainer(node, clause, multiLine); } updateImportAttributes(node: ts.ImportAttributes, elements: ts.NodeArray<ts.ImportAttribute>, multiLine?: boolean): ts.ImportAttributes { return ts.factory.updateImportAttributes(node, elements, multiLine); } updateImportAttribute(node: ts.ImportAttribute, name: string, value: ts.Expression): ts.ImportAttribute { return ts.factory.updateImportAttribute(node, this.createIdentifier(name)!, value); } updateNamespaceImport(node: ts.NamespaceImport, name: string): ts.NamespaceImport { return ts.factory.updateNamespaceImport(node, this.createIdentifier(name)!); } updateNamespaceExport(node: ts.NamespaceExport, name: string): ts.NamespaceExport { return ts.factory.updateNamespaceExport(node, this.createIdentifier(name)!); } updateNamedImports(node: ts.NamedImports, ...elements: ts.ImportSpecifier[]): ts.NamedImports { return ts.factory.updateNamedImports(node, elements); } // Additional methods to cover common usages in NodeClassFactory.ts createImportDeclaration(moduleSpecifier: string, namedImports: string[], isTypeOnly = false, importAttributes?: ts.ImportAttributes, ...modifierFlags: ModifierFlags[]): ts.ImportDeclaration { return ts.factory.createImportDeclaration( this.createModifiers(...modifierFlags), this.createImportClause(isTypeOnly, undefined, ts.factory.createNamedImports(namedImports.map((name) => ts.factory.createImportSpecifier(false, undefined, this.createIdentifier(name)!)))), ts.factory.createStringLiteral(moduleSpecifier), importAttributes ); } createVariableStatement(name: string, initializer: ts.Expression, isConst = true): ts.VariableStatement { return ts.factory.createVariableStatement( [ts.factory.createModifier(ts.SyntaxKind.ExportKeyword)], ts.factory.createVariableDeclarationList( [ts.factory.createVariableDeclaration(this.createIdentifier(name)!, undefined, undefined, initializer)], isConst ? ts.NodeFlags.Const : ts.NodeFlags.None ) ); } createTypeAliasDeclaration(name: string, type: ts.TypeNode): ts.TypeAliasDeclaration { return ts.factory.createTypeAliasDeclaration(undefined, this.createIdentifier(name)!, undefined, type); } createClassDeclaration(name: string, members: ts.ClassElement[], heritageClauses?: ts.HeritageClause[]): ts.ClassDeclaration { return ts.factory.createClassDeclaration(this.createModifiers(ts.ModifierFlags.Export), this.createIdentifier(name)!, undefined, heritageClauses, members); } and<T>(...flags: T[]): T { if (flags.length === 0) { return 0 as T; } return flags.reduce((acc: number, flag) => acc & (flag as number), flags[0] as number) as T; } or<T>(...flags: T[]): T { if (flags.length === 0) { return 0 as T; } return flags.reduce((acc: number, flag) => acc | (flag as number), flags[0] as number) as T; } createModifiers(...flags: ts.ModifierFlags[]): ts.Modifier[] { return ts.factory.createModifiersFromModifierFlags(this.or(...flags)); } createMethodDeclaration(name: string, parameters: ts.ParameterDeclaration[], body: ts.Block, isAsync = false): ts.MethodDeclaration { return ts.factory.createMethodDeclaration( isAsync ? this.createModifiers(ts.ModifierFlags.Async) : undefined, undefined, this.createIdentifier(name)!, undefined, undefined, parameters, undefined, body ); } createPropertyDeclaration(name: string, initializer: ts.Expression, modifiers: ts.Modifier[] = []): ts.PropertyDeclaration { return ts.factory.createPropertyDeclaration(modifiers, this.createIdentifier(name)!, undefined, undefined, initializer); } createConstructorDeclaration(parameters: ts.ParameterDeclaration[], body: ts.Block): ts.ConstructorDeclaration { return ts.factory.createConstructorDeclaration(undefined, parameters, body); } createFunctionDeclaration(name: string, parameters: ts.ParameterDeclaration[], body: ts.Block, returnType?: ts.TypeNode): ts.FunctionDeclaration { return ts.factory.createFunctionDeclaration([this.createModifier(ts.SyntaxKind.ExportKeyword)], undefined, this.createIdentifier(name)!, undefined, parameters, returnType, body); } createModifier(ExportKeyword: ts.ModifierSyntaxKind): ts.ModifierLike { return ts.factory.createModifier(ExportKeyword); } createModuleDeclaration(name: string, body: ts.ModuleBlock, ...modifierFlags: ModifierFlags[]): ts.ModuleDeclaration { return ts.factory.createModuleDeclaration(this.createModifiers(...modifierFlags), this.createIdentifier(name)!, body, ts.NodeFlags.Namespace); } createToken<T extends ModifierSyntaxKind>(kind: T): ts.ModifierLike; createToken<T extends PunctuationSyntaxKind>(kind: T): ts.PunctuationToken<T>; createToken<T extends TokenSyntaxKind>(kind: T): ts.Token<T> { return ts.factory.createToken(kind); } createPropertySignature(name: string, type: ts.TypeNode, optional = false, ...modifiers: ModifierFlags[]): ts.PropertySignature { return ts.factory.createPropertySignature(this.createModifiers(...modifiers), this.createIdentifier(name)!, optional ? this.createToken(ts.SyntaxKind.QuestionToken) : undefined, type); } createTypeReferenceNode(typeName: string, qualifier: string): ts.TypeReferenceNode; createTypeReferenceNode(typeName: string, ...typeArguments: ts.TypeNode[]): ts.TypeReferenceNode; createTypeReferenceNode(typeName: string, qualifier?: string | ts.TypeNode, ...typeArguments: ts.TypeNode[]): ts.TypeReferenceNode { if (typeof qualifier === 'string') { return ts.factory.createTypeReferenceNode(ts.factory.createQualifiedName(this.createIdentifier(qualifier), this.createIdentifier(typeName)!), typeArguments); } else { return ts.factory.createTypeReferenceNode(this.createIdentifier(typeName)!, [qualifier, ...typeArguments]); } } typesEqual(a: ts.TypeNode, b: ts.TypeNode): boolean { if (ts.isLiteralTypeNode(a) && ts.isLiteralTypeNode(b)) { if (ts.isNumericLiteral(a.literal) && ts.isNumericLiteral(b.literal)) { return a.literal.text === b.literal.text; } else if (ts.isStringLiteral(a.literal) && ts.isStringLiteral(b.literal)) { return a.literal.text === b.literal.text; } else if (ts.isBigIntLiteral(a.literal) && ts.isBigIntLiteral(b.literal)) { return a.literal.text === b.literal.text; } } if (ts.isTypeReferenceNode(a) && ts.isTypeReferenceNode(b)) { if (ts.isIdentifier(a.typeName) && ts.isIdentifier(b.typeName)) { return a.typeName.text === b.typeName.text; } } if(a.kind === ts.SyntaxKind.NumberKeyword || a.kind === ts.SyntaxKind.StringKeyword || a.kind === ts.SyntaxKind.BigIntKeyword) { return a.kind === b.kind; } return false; } createUnionTypeNode(...types: ts.TypeNode[]): ts.UnionTypeNode { let t = unique(types, this.typesEqual); /*try { let t = [types[0]]; for (let i = 1; i < types.length; i++) { if (types[i].kind !== ts.SyntaxKind.NeverKeyword) { if (types[i].kind === ts.SyntaxKind.UnionType) { types.push(...(types[i] as ts.UnionTypeNode).types); } else if (!t.some((x) => this.typesEqual(x, types[i]))) t.push(types[i]); } } return ts.factory.createUnionTypeNode(t); } catch(e) {*/ //console.warn(e); return ts.factory.createUnionTypeNode(t); //} } createIntersectionTypeNode(types: ts.TypeNode[]): ts.IntersectionTypeNode { return ts.factory.createIntersectionTypeNode(types); } createTypeLiteralNode(members: ts.TypeElement[]): ts.TypeLiteralNode { return ts.factory.createTypeLiteralNode(members); } createFunctionTypeNode(type: ts.TypeNode, typeParameters: readonly ts.TypeParameterDeclaration[], ...parameters: ts.ParameterDeclaration[]): ts.FunctionTypeNode { return ts.factory.createFunctionTypeNode(typeParameters, parameters, type); } createParameter(name: string, type: ts.TypeNode, initializer?: ts.Expression, optional = false): ts.ParameterDeclaration { return ts.factory.createParameterDeclaration(undefined, undefined, this.createIdentifier(name)!, optional ? this.createToken(ts.SyntaxKind.QuestionToken) : undefined, type, initializer); } createExpressionStatement(expression: ts.Expression): ts.ExpressionStatement { return ts.factory.createExpressionStatement(expression); } createCallExpression(expression: ts.Expression, typeArguments: ts.TypeNode[] | undefined, argumentsArray: ts.Expression[]): ts.CallExpression { return ts.factory.createCallExpression(expression, typeArguments, argumentsArray); } createReturnStatement(expression?: ts.Expression): ts.ReturnStatement { return ts.factory.createReturnStatement(expression); } createNewExpression(expression: ts.Expression, typeArguments: ts.TypeNode[] | undefined, argumentsArray: ts.Expression[]): ts.NewExpression { return ts.factory.createNewExpression(expression, typeArguments, argumentsArray); } createObjectLiteral(properties: ts.ObjectLiteralElementLike[], multiLine?: boolean): ts.ObjectLiteralExpression { return ts.factory.createObjectLiteralExpression(properties, multiLine); } createPropertyAssignment(name: string, initializer: ts.Expression): ts.PropertyAssignment { return ts.factory.createPropertyAssignment(this.createIdentifier(name)!, initializer); } createQualifiedName(...names: (ts.EntityName | string)[]): ts.EntityName { let n = names.pop(); if (!n) throw new Error('No names provided'); let name = typeof n === 'string' ? this.createIdentifier(n) : n; if(names.length === 0) { return name; } else { return ts.factory.createQualifiedName(this.createQualifiedName(...names), name as ts.Identifier); } } createLiteralTypeNode(literal: string | number): ts.LiteralTypeNode { if (typeof literal === 'number') { return ts.factory.createLiteralTypeNode(this.createNumericLiteral(literal as number)); } return ts.factory.createLiteralTypeNode(this.createStringLiteral(literal)); } createLiteral(value: string | number): ts.LiteralExpression | ts.PrefixUnaryExpression { if (typeof value === 'number') { return this.createNumericLiteral(value as number); } return this.createStringLiteral(value); } createNumericLiteral(value: number, flags?: ts.TokenFlags): ts.NumericLiteral | ts.PrefixUnaryExpression{ if(value < 0) { return ts.factory.createPrefixUnaryExpression(SyntaxKind.MinusToken, ts.factory.createNumericLiteral((-value).toString(), flags)); } return ts.factory.createNumericLiteral(value.toString(), flags); } createStringLiteral(text: string): ts.StringLiteral { return ts.factory.createStringLiteral(text, true); } createKeywordTypeNode(kind: ts.KeywordTypeSyntaxKind): ts.KeywordTypeNode { return ts.factory.createKeywordTypeNode(kind); } createTypePredicateNode(parameterName: ts.Identifier | string, type: ts.TypeNode): ts.TypePredicateNode { return ts.factory.createTypePredicateNode(undefined, parameterName, type); } createBinaryExpression(left: ts.Expression, operator: ts.BinaryOperator, right: ts.Expression): ts.BinaryExpression { return ts.factory.createBinaryExpression(left, operator, right); } createPropertyAccessExpression(expression: ts.Expression, name: string | ts.MemberName): ts.PropertyAccessExpression { return ts.factory.createPropertyAccessExpression(expression, name); } createHeritageClause(token: ts.SyntaxKind.ExtendsKeyword | ts.SyntaxKind.ImplementsKeyword, ...types: ts.ExpressionWithTypeArguments[]): ts.HeritageClause { return ts.factory.createHeritageClause(token, types); } createExpressionWithTypeArguments(expression: ts.Expression, ...typeArguments: ts.TypeNode[]): ts.ExpressionWithTypeArguments { return ts.factory.createExpressionWithTypeArguments(expression, typeArguments); } createBlock(multiLine: boolean, ...statements: ts.Statement[]): ts.Block; createBlock(...statements: ts.Statement[]); createBlock(multiLine: boolean | ts.Statement, ...statements: ts.Statement[]): ts.Block { if(typeof multiLine === 'boolean') { return ts.factory.createBlock(statements, multiLine); } return ts.factory.createBlock([multiLine, ...statements],true); } createModuleBlock(...statements: ts.Statement[]): ts.ModuleBlock { return ts.factory.createModuleBlock(statements); } createInterfaceDeclaration(name: string, members: ts.TypeElement[], heritageClauses?: ts.HeritageClause[]): ts.InterfaceDeclaration { return ts.factory.createInterfaceDeclaration([ts.factory.createModifier(ts.SyntaxKind.ExportKeyword)], this.createIdentifier(name)!, undefined, heritageClauses, members); } createTypeOperatorNode(operator: ts.SyntaxKind.KeyOfKeyword | ts.SyntaxKind.ReadonlyKeyword | ts.SyntaxKind.UniqueKeyword, type: ts.TypeNode): ts.TypeOperatorNode { return ts.factory.createTypeOperatorNode(operator, type); } createTypeQueryNode(exprName: ts.EntityName): ts.TypeQueryNode { return ts.factory.createTypeQueryNode(exprName); } createAsExpression(expression: ts.Expression, type: ts.TypeNode): ts.AsExpression { return ts.factory.createAsExpression(expression, type); } createPropertyAccessChain(name: string, useQuestionDot: boolean = true, expression: ts.Expression): ts.PropertyAccessChain { return ts.factory.createPropertyAccessChain(expression, useQuestionDot ? this.createToken(ts.SyntaxKind.QuestionDotToken) : undefined, this.createIdentifier(name)!); } }