langium
Version:
A language engineering tool for the Language Server Protocol
240 lines (219 loc) • 9.3 kB
text/typescript
/******************************************************************************
* Copyright 2022 TypeFox GmbH
* This program and the accompanying materials are made available under the
* terms of the MIT License, which is available in the project root.
******************************************************************************/
import type { References } from '../../references/references.js';
import type { AstNodeLocator } from '../../workspace/ast-node-locator.js';
import type { LangiumDocuments } from '../../workspace/documents.js';
import type { Interface, Type, AbstractType, TypeDefinition } from '../../languages/generated/ast.js';
import type { PlainInterface, PlainProperty } from './type-collector/plain-types.js';
import type { AstTypes, InterfaceType, PropertyType, TypeOption } from './type-collector/types.js';
import { MultiMap } from '../../utils/collections.js';
import { isInterface, isType, isUnionType, isSimpleType } from '../../languages/generated/ast.js';
import { isArrayType, isPrimitiveType, isPropertyUnion, isReferenceType, isValueType } from './type-collector/types.js';
/**
* Collects all properties of all interface types. Includes super type properties.
* @param interfaces A topologically sorted array of interfaces.
*/
export function collectAllPlainProperties(interfaces: PlainInterface[]): MultiMap<string, PlainProperty> {
const map = new MultiMap<string, PlainProperty>();
for (const interfaceType of interfaces) {
map.addAll(interfaceType.name, interfaceType.properties);
}
for (const interfaceType of interfaces) {
for (const superType of interfaceType.superTypes) {
const superTypeProperties = map.get(superType);
if (superTypeProperties) {
map.addAll(interfaceType.name, superTypeProperties);
}
}
}
return map;
}
export function distinctAndSorted<T>(list: T[], compareFn?: (a: T, b: T) => number): T[] {
return Array.from(new Set(list)).sort(compareFn);
}
export function collectChildrenTypes(interfaceNode: Interface, references: References, langiumDocuments: LangiumDocuments, nodeLocator: AstNodeLocator): Set<Interface | Type> {
const childrenTypes = new Set<Interface | Type>();
childrenTypes.add(interfaceNode);
const refs = references.findReferences(interfaceNode, {});
for (const ref of refs) {
const doc = langiumDocuments.getDocument(ref.sourceUri);
if (!doc) {
continue;
}
const astNode = nodeLocator.getAstNode(doc.parseResult.value, ref.sourcePath);
if (isInterface(astNode)) {
childrenTypes.add(astNode);
const childrenOfInterface = collectChildrenTypes(astNode, references, langiumDocuments, nodeLocator);
childrenOfInterface.forEach(child => childrenTypes.add(child));
} else if (astNode && isType(astNode.$container)) {
childrenTypes.add(astNode.$container);
}
}
return childrenTypes;
}
export function collectTypeHierarchy(types: TypeOption[]): {
superTypes: MultiMap<string, string>
subTypes: MultiMap<string, string>
} {
const allTypes = new Set(types);
const duplicateSuperTypes = new MultiMap<string, string>();
const duplicateSubTypes = new MultiMap<string, string>();
for (const type of allTypes) {
for (const superType of type.superTypes) {
if (allTypes.has(superType)) {
duplicateSuperTypes.add(type.name, superType.name);
duplicateSubTypes.add(superType.name, type.name);
}
}
for (const subType of type.subTypes) {
if (allTypes.has(subType)) {
duplicateSuperTypes.add(subType.name, type.name);
duplicateSubTypes.add(type.name, subType.name);
}
}
}
const superTypes = new MultiMap<string, string>();
const subTypes = new MultiMap<string, string>();
// Deduplicate and sort
for (const [name, superTypeList] of Array.from(duplicateSuperTypes.entriesGroupedByKey()).sort(([aName], [bName]) => aName.localeCompare(bName))) {
superTypes.addAll(name, Array.from(new Set(superTypeList)));
}
for (const [name, subTypeList] of Array.from(duplicateSubTypes.entriesGroupedByKey()).sort(([aName], [bName]) => aName.localeCompare(bName))) {
subTypes.addAll(name, Array.from(new Set(subTypeList)));
}
return {
superTypes,
subTypes
};
}
export function collectSuperTypes(ruleNode: AbstractType): Set<Interface> {
const superTypes = new Set<Interface>();
if (isInterface(ruleNode)) {
superTypes.add(ruleNode);
ruleNode.superTypes.forEach(superType => {
if (isInterface(superType.ref)) {
superTypes.add(superType.ref);
const collectedSuperTypes = collectSuperTypes(superType.ref);
for (const superType of collectedSuperTypes) {
superTypes.add(superType);
}
}
});
} else if (isType(ruleNode)) {
const usedTypes = collectUsedTypes(ruleNode.type);
for (const usedType of usedTypes) {
const collectedSuperTypes = collectSuperTypes(usedType);
for (const superType of collectedSuperTypes) {
superTypes.add(superType);
}
}
}
return superTypes;
}
function collectUsedTypes(typeDefinition: TypeDefinition): Array<Interface | Type> {
if (isUnionType(typeDefinition)) {
return typeDefinition.types.flatMap(e => collectUsedTypes(e));
} else if (isSimpleType(typeDefinition)) {
const value = typeDefinition.typeRef?.ref;
if (isType(value) || isInterface(value)) {
return [value];
}
}
return [];
}
export function mergeInterfaces(inferred: AstTypes, declared: AstTypes): InterfaceType[] {
return inferred.interfaces.concat(declared.interfaces);
}
export function mergeTypesAndInterfaces(astTypes: AstTypes): TypeOption[] {
return (astTypes.interfaces as TypeOption[]).concat(astTypes.unions);
}
export function hasArrayType(type: PropertyType): boolean {
if (isPropertyUnion(type)) {
return type.types.some(e => hasArrayType(e));
} else if (isArrayType(type)) {
return true;
} else {
return false;
}
}
export function hasBooleanType(type: PropertyType): boolean {
if (isPropertyUnion(type)) {
return type.types.some(e => hasBooleanType(e));
} else if (isPrimitiveType(type)) {
return type.primitive === 'boolean';
} else {
return false;
}
}
export function findReferenceTypes(type: PropertyType): string[] {
if (isPropertyUnion(type)) {
return type.types.flatMap(e => findReferenceTypes(e));
} else if (isReferenceType(type)) {
const refType = type.referenceType;
if (isValueType(refType)) {
return [refType.value.name];
}
} else if (isArrayType(type)) {
return type.elementType ? findReferenceTypes(type.elementType) : [];
}
return [];
}
export function findAstTypes(type: PropertyType): string[] {
return findAstTypesInternal(type, new Set());
}
function findAstTypesInternal(type: PropertyType, visited: Set<PropertyType>): string[] {
if (visited.has(type)) {
return [];
} else {
visited.add(type);
}
if (isPropertyUnion(type)) {
return type.types.flatMap(e => findAstTypesInternal(e, visited));
} else if (isValueType(type)) {
const value = type.value;
if ('type' in value) {
return findAstTypesInternal(value.type, visited);
} else {
return [value.name];
}
} else if (isArrayType(type)) {
return type.elementType ? findAstTypesInternal(type.elementType, visited) : [];
}
return [];
}
export function isAstType(type: PropertyType): boolean {
return isAstTypeInternal(type, new Map());
}
export function isAstTypeInternal(type: PropertyType, visited: Map<PropertyType, boolean>): boolean {
if (visited.has(type)) {
return visited.get(type)!;
}
// This is supposed to prevent infinite recursion.
// Setting this to true is a pretty safe assumption.
// Setting it to false might lead to false negatives for property unions.
visited.set(type, true);
let result = false;
if (isPropertyUnion(type)) {
result = type.types.every(e => isAstTypeInternal(e, visited));
} else if (isValueType(type)) {
const value = type.value;
if ('type' in value) {
result = isAstTypeInternal(value.type, visited);
} else {
// Is definitely an interface type
result = true;
}
}
visited.set(type, result);
return result;
}
export function escapeQuotes(str: string, type: '"' | "'" = '"'): string {
if (type === '"') {
return str.replace(/"/g, '\\"');
} else {
return str.replace(/'/g, "\\'");
}
}