/* eslint-disable @typescript-eslint/no-loop-func */ // Copyright (c) Microsoft Corporation. All rights reserved. Licensed under the MIT license. // See LICENSE in the project root for license information. import { type DocNode, type DocPlainText, DocDeclarationReference, DocNodeKind, TSDocConfiguration, DocMemberReference, DocMemberIdentifier, } from '@microsoft/tsdoc'; import type { DeclarationReference } from '@microsoft/tsdoc/lib-commonjs/beta/DeclarationReference.js'; import { InternalError } from '@rushstack/node-core-library'; import type { IExcerptToken, IExcerptTokenRange } from '../index.js'; import { ApiDeclaredItem } from '../index.js'; import type { IApiDeclaredItemJson } from '../items/ApiDeclaredItem.js'; import { ApiItem, apiItem_onParentChanged, type IApiItemJson, type IApiItemOptions, type IApiItemConstructor, ApiItemKind, } from '../items/ApiItem.js'; import type { ApiClass } from '../model/ApiClass.js'; import type { ApiInterface } from '../model/ApiInterface.js'; import type { ApiModel } from '../model/ApiModel.js'; import { ApiJsonSchemaVersion, type DeserializerContext } from '../model/DeserializerContext.js'; import type { HeritageType } from '../model/HeritageType.js'; import type { IResolveDeclarationReferenceResult } from '../model/ModelReferenceResolver.js'; import { ApiNameMixin } from './ApiNameMixin.js'; import type { ExcerptToken } from './Excerpt.js'; import { ExcerptTokenKind } from './Excerpt.js'; import { type IFindApiItemsResult, type IFindApiItemsMessage, FindApiItemsMessageId } from './IFindApiItemsResult.js'; /** * Constructor options for {@link (ApiItemContainerMixin:interface)}. * * @public */ export interface IApiItemContainerMixinOptions extends IApiItemOptions { members?: ApiItem[] | undefined; preserveMemberOrder?: boolean | undefined; } export interface IApiItemContainerJson extends IApiItemJson { members: IApiItemJson[]; preserveMemberOrder?: boolean; } interface Mixin { declarationReference: DocDeclarationReference; typeParameters: IExcerptTokenRange[]; } interface ExcerptTokenRangeInDeclaredItem { item: ApiDeclaredItem; range: IExcerptTokenRange; } interface IMappedTypeParameters { item: ApiItem; mappedTypeParameters: Map; } const _members: unique symbol = Symbol('ApiItemContainerMixin._members'); const _membersSorted: unique symbol = Symbol('ApiItemContainerMixin._membersSorted'); const _membersByContainerKey: unique symbol = Symbol('ApiItemContainerMixin._membersByContainerKey'); const _membersByName: unique symbol = Symbol('ApiItemContainerMixin._membersByName'); const _membersByKind: unique symbol = Symbol('ApiItemContainerMixin._membersByKind'); const _preserveMemberOrder: unique symbol = Symbol('ApiItemContainerMixin._preserveMemberOrder'); /** * The mixin base class for API items that act as containers for other child items. * * @remarks * * This is part of the {@link ApiModel} hierarchy of classes, which are serializable representations of * API declarations. The non-abstract classes (e.g. `ApiClass`, `ApiEnum`, `ApiInterface`, etc.) use * TypeScript "mixin" functions (e.g. `ApiDeclaredItem`, `ApiItemContainerMixin`, etc.) to add various * features that cannot be represented as a normal inheritance chain (since TypeScript does not allow a child class * to extend more than one base class). The "mixin" is a TypeScript merged declaration with three components: * the function that generates a subclass, an interface that describes the members of the subclass, and * a namespace containing static members of the class. * * Examples of `ApiItemContainerMixin` child classes include `ApiModel`, `ApiPackage`, `ApiEntryPoint`, * and `ApiEnum`. But note that `Parameter` is not considered a "member" of an `ApiMethod`; this relationship * is modeled using {@link (ApiParameterListMixin:interface).parameters} instead * of {@link ApiItem.members}. * @public */ export interface ApiItemContainerMixin extends ApiItem { /** * For a given member of this container, return its `ApiItem.getMergedSiblings()` list. * * @internal */ _getMergedSiblingsForMember(memberApiItem: ApiItem): readonly ApiItem[]; /** * Adds a new member to the container. * * @remarks * An ApiItem cannot be added to more than one container. */ addMember(member: ApiItem): void; /** * Returns a list of members with the specified name. */ findMembersByName(name: string): readonly ApiItem[]; /** * Finds all of the ApiItem's immediate and inherited members by walking up the inheritance tree. * * @remarks * * Given the following class heritage: * * ``` * export class A { * public a: number|boolean; * } * * export class B extends A { * public a: number; * public b: string; * } * * export class C extends B { * public c: boolean; * } * ``` * * Calling `findMembersWithInheritance` on `C` will return `B.a`, `B.b`, and `C.c`. Calling the * method on `B` will return `B.a` and `B.b`. And calling the method on `A` will return just * `A.a`. * * The inherited members returned by this method may be incomplete. If so, there will be a flag * on the result object indicating this as well as messages explaining the errors in more detail. * Some scenarios include: * * - Interface extending from a type alias. * * - Class extending from a variable. * * - Extending from a declaration not present in the model (e.g. external package). * * - Extending from an unexported declaration (e.g. ae-forgotten-export). Common in mixin * patterns. * * - Unexpected runtime errors... * * Lastly, be aware that the types of inherited members are returned with respect to their * defining class as opposed to with respect to the inheriting class. For example, consider * the following: * * ``` * export class A { * public a: T; * } * * export class B extends A {} * ``` * * When called on `B`, this method will return `B.a` with type `T` as opposed to type * `number`, although the latter is more accurate. */ findMembersWithInheritance(): IFindApiItemsResult; /** * Disables automatic sorting of {@link ApiItem.members}. * * @remarks * By default `ApiItemContainerMixin` will automatically sort its members according to their * {@link ApiItem.getSortKey} string, which provides a standardized mostly alphabetical ordering * that is appropriate for most API items. When loading older .api.json files the automatic sorting * is reapplied and may update the ordering. * * Set `preserveMemberOrder` to true to disable automatic sorting for this container; instead, the * members will retain whatever ordering appeared in the {@link IApiItemContainerMixinOptions.members} array. * The `preserveMemberOrder` option is saved in the .api.json file. */ readonly preserveMemberOrder: boolean; /** * @override */ serializeInto(jsonObject: Partial): void; /** * Attempts to retrieve a member using its containerKey, or returns `undefined` if no matching member was found. * * @remarks * Use the `getContainerKey()` static member to construct the key. Each subclass has a different implementation * of this function, according to the aspects that are important for identifying it. * * See {@link ApiItem.containerKey} for more information. */ tryGetMemberByKey(containerKey: string): ApiItem | undefined; } /** * Mixin function for {@link ApiDeclaredItem}. * * @param baseClass - The base class to be extended * @returns A child class that extends baseClass, adding the {@link (ApiItemContainerMixin:interface)} functionality. * @public */ export function ApiItemContainerMixin( baseClass: TBaseClass, ): TBaseClass & (new (...args: any[]) => ApiItemContainerMixin) { class MixedClass extends baseClass implements ApiItemContainerMixin { public readonly [_members]: ApiItem[]; public [_membersSorted]: boolean; public [_membersByContainerKey]: Map; public [_preserveMemberOrder]: boolean; // For members of this container that extend ApiNameMixin, this stores the list of members with a given name. // Examples include merged declarations, overloaded functions, etc. public [_membersByName]: Map | undefined; // For members of this container that do NOT extend ApiNameMixin, this stores the list of members // that share a common ApiItemKind. Examples include overloaded constructors or index signatures. public [_membersByKind]: Map | undefined; // key is ApiItemKind public constructor(...args: any[]) { super(...args); const options: IApiItemContainerMixinOptions = args[0] as IApiItemContainerMixinOptions; this[_members] = []; this[_membersSorted] = false; this[_membersByContainerKey] = new Map(); this[_preserveMemberOrder] = options.preserveMemberOrder ?? false; if (options.members) { for (const member of options.members) { this.addMember(member); } } } /** * @override */ public static override onDeserializeInto( options: Partial, context: DeserializerContext, jsonObject: IApiItemContainerJson, ): void { baseClass.onDeserializeInto(options, context, jsonObject); options.preserveMemberOrder = jsonObject.preserveMemberOrder; options.members = []; for (const memberObject of jsonObject.members) { options.members.push(ApiItem.deserialize(memberObject, context)); } } /** * @override */ public override get members(): readonly ApiItem[] { if (!this[_membersSorted] && !this[_preserveMemberOrder]) { this[_members].sort((x, y) => x.getSortKey().localeCompare(y.getSortKey())); this[_membersSorted] = true; } return this[_members]; } public get preserveMemberOrder(): boolean { return this[_preserveMemberOrder]; } public addMember(member: ApiItem): void { if (this[_membersByContainerKey].has(member.containerKey)) { throw new Error( `Another member has already been added with the same name (${member.displayName})` + ` and containerKey (${member.containerKey})`, ); } const existingParent: ApiItem | undefined = member.parent; if (existingParent !== undefined) { throw new Error(`This item has already been added to another container: "${existingParent.displayName}"`); } this[_members].push(member); this[_membersByName] = undefined; // invalidate the lookup this[_membersByKind] = undefined; // invalidate the lookup this[_membersSorted] = false; this[_membersByContainerKey].set(member.containerKey, member); member[apiItem_onParentChanged](this); } public tryGetMemberByKey(containerKey: string): ApiItem | undefined { return this[_membersByContainerKey].get(containerKey); } public findMembersByName(name: string): readonly ApiItem[] { this._ensureMemberMaps(); return this[_membersByName]!.get(name) ?? []; } public findMembersWithInheritance(): IFindApiItemsResult { const messages: IFindApiItemsMessage[] = []; let maybeIncompleteResult = false; // For API items that don't support inheritance, this method just returns the item's // immediate members. switch (this.kind) { case ApiItemKind.Class: case ApiItemKind.Interface: break; default: { return { items: this.members.concat(), messages, maybeIncompleteResult, }; } } // The Deserializer class is coupled with a ton of other classes, so we delay loading it // to avoid ES5 circular imports. // *eslint-disable-next-line @typescript-eslint/consistent-type-imports, @typescript-eslint/no-require-imports, @typescript-eslint/no-var-requires // const deserializerModule: typeof import('../model/Deserializer') = require('../model/Deserializer'); const membersByName: Map = new Map(); const membersByKind: Map = new Map(); const toVisit: IMappedTypeParameters[] = []; let next: IMappedTypeParameters | undefined = { item: this, mappedTypeParameters: new Map() }; while (next?.item) { const membersToAdd: ApiItem[] = []; //* const typeParams = next.mappedTypeParameters; const context: DeserializerContext = { apiJsonFilename: '', toolPackage: '', toolVersion: '', versionToDeserialize: ApiJsonSchemaVersion.LATEST, tsdocConfiguration: new TSDocConfiguration(), }; // */ // eslint-disable-next-line @typescript-eslint/consistent-type-imports, @typescript-eslint/no-require-imports, @typescript-eslint/no-var-requires const deserializerModule: typeof import('../model/Deserializer') = require('../model/Deserializer'); // For each member, check to see if we've already seen a member with the same name // previously in the inheritance tree. If so, we know we won't inherit it, and thus // do not add it to our `membersToAdd` array. for (let member of next.item.members) { // We add the to-be-added members to an intermediate array instead of immediately // to the maps themselves to support method overloads with the same name. // This was supposed to replace type parameters with their assigned values in inheritance, but doesn't work yet //* if (member instanceof ApiDeclaredItem && member.excerptTokens.some((token) => typeParams.has(token.text))) { const jsonObject: Partial = {}; member.serializeInto(jsonObject); const excerptTokens = (jsonObject as IApiDeclaredItemJson).excerptTokens.map((token) => { let x: ExcerptToken | undefined; if (typeParams.has(token.text) && next?.item instanceof ApiDeclaredItem) { const originalValue = typeParams.get(token.text)!; x = originalValue.item.excerptTokens[originalValue.range.startIndex]; } const excerptToken: IExcerptToken = x ? { kind: x.kind, text: x.text } : token; if (x?.canonicalReference !== undefined) { excerptToken.canonicalReference = x.canonicalReference.toString(); } return excerptToken; }); member = deserializerModule.Deserializer.deserialize(context, { ...jsonObject, excerptTokens, } as IApiDeclaredItemJson); member[apiItem_onParentChanged](next.item); } if (ApiNameMixin.isBaseClassOf(member)) { if (!membersByName.has(member.name)) { membersToAdd.push(member); } } else if (!membersByKind.has(member.kind)) { membersToAdd.push(member); } } for (const member of membersToAdd) { if (ApiNameMixin.isBaseClassOf(member)) { const members: ApiItem[] = membersByName.get(member.name) ?? []; members.push(member); membersByName.set(member.name, members); } else { const members: ApiItem[] = membersByKind.get(member.kind) ?? []; members.push(member); membersByKind.set(member.kind, members); } } const findPlainTextNode = (node: DocNode): DocPlainText | undefined => { switch (node.kind) { case DocNodeKind.PlainText: return node as DocPlainText; default: for (const child of node.getChildNodes()) { const result = findPlainTextNode(child); if (result) return result; } } return undefined; }; // Interfaces can extend multiple interfaces, so iterate through all of them. // Also Classes can have multiple mixins const extendedItems: IMappedTypeParameters[] = []; let extendsTypes: readonly (HeritageType | Mixin)[] | undefined; switch (next.item.kind) { case ApiItemKind.Class: { const apiClass: ApiClass = next.item as ApiClass; const configuration = apiClass.tsdocComment?.configuration ?? new TSDocConfiguration(); const mixins = apiClass.tsdocComment?.customBlocks .filter( (block) => block.blockTag.tagName === '@mixes', // && // block.getChildNodes().some((node) => node.kind === DocNodeKind.PlainText), ) .map(findPlainTextNode) .filter((block) => block !== undefined) .map((block) => ({ declarationReference: new DocDeclarationReference({ configuration, memberReferences: block.text.split('.').map( (part, index) => new DocMemberReference({ configuration, hasDot: index > 0, memberIdentifier: new DocMemberIdentifier({ configuration, identifier: part }), }), ), }), typeParameters: [] as IExcerptTokenRange[], })) ?? []; extendsTypes = apiClass.extendsType ? [apiClass.extendsType, ...mixins] : [...mixins]; break; } case ApiItemKind.Interface: { const apiInterface: ApiInterface = next.item as ApiInterface; extendsTypes = apiInterface.extendsTypes; break; } default: break; } if (extendsTypes === undefined) { messages.push({ messageId: FindApiItemsMessageId.UnsupportedKind, text: `Unable to analyze references of API item ${next.item.displayName} because it is of unsupported kind ${next.item.kind}`, }); maybeIncompleteResult = true; next = toVisit.shift(); continue; } for (const extendsType of extendsTypes) { let canonicalReference: DeclarationReference | DocDeclarationReference; if ('excerpt' in extendsType) { // We want to find the reference token associated with the actual inherited declaration. // In every case we support, this is the first reference token. For example: // // ``` // export class A extends B {} // ^ // export class A extends B {} // ^ // export class A extends B.C {} // ^^^ // ``` const firstReferenceToken: ExcerptToken | undefined = extendsType.excerpt.spannedTokens.find( (token: ExcerptToken) => { return token.kind === ExcerptTokenKind.Reference && token.canonicalReference; }, ); if (!firstReferenceToken) { messages.push({ messageId: FindApiItemsMessageId.ExtendsClauseMissingReference, text: `Unable to analyze extends clause ${extendsType.excerpt.text} of API item ${next.item.displayName} because no canonical reference was found`, }); maybeIncompleteResult = true; continue; } canonicalReference = firstReferenceToken.canonicalReference!; } else { // extendsType is a Mixin canonicalReference = extendsType.declarationReference; } const apiModel: ApiModel | undefined = this.getAssociatedModel(); if (!apiModel) { messages.push({ messageId: FindApiItemsMessageId.NoAssociatedApiModel, text: `Unable to analyze references of API item ${next.item.displayName} because it is not associated with an ApiModel`, }); maybeIncompleteResult = true; continue; } const apiItemResult: IResolveDeclarationReferenceResult = apiModel.resolveDeclarationReference( canonicalReference, this, ); const apiItem: ApiItem | undefined = apiItemResult.resolvedApiItem; if (!apiItem) { messages.push({ messageId: FindApiItemsMessageId.DeclarationResolutionFailed, text: `Unable to resolve declaration reference within API item ${next.item.displayName}: ${apiItemResult.errorMessage}`, }); maybeIncompleteResult = true; continue; } const mappedTypeParameters: Map = new Map(); if ( (apiItem.kind === ApiItemKind.Class || apiItem.kind === ApiItemKind.Interface) && next.item.kind === ApiItemKind.Class ) { for (const [index, key] of (apiItem as ApiClass | ApiInterface).typeParameters.entries() ?? []) { const typeParameter = extendsType.typeParameters?.[index]; if (typeParameter) mappedTypeParameters.set(key.name, { item: next.item as ApiDeclaredItem, range: typeParameter }); else if (key.defaultTypeExcerpt) mappedTypeParameters.set(key.name, { item: apiItem as ApiDeclaredItem, range: key.defaultTypeExcerpt.tokenRange, }); } } extendedItems.push({ item: apiItem, mappedTypeParameters }); } // For classes, this array will only have one item. For interfaces, there may be multiple items. Sort the array // into alphabetical order before adding to our list of API items to visit. This ensures that in the case // of multiple interface inheritance, a member inherited from multiple interfaces is attributed to the interface // earlier in alphabetical order (as opposed to source order). // // For example, in the code block below, `Bar.x` is reported as the inherited item, not `Foo.x`. // // ``` // interface Foo { // public x: string; // } // // interface Bar { // public x: string; // } // // interface FooBar extends Foo, Bar {} // ``` extendedItems.sort((x: IMappedTypeParameters, y: IMappedTypeParameters) => x.item.getSortKey().localeCompare(y.item.getSortKey()), ); toVisit.push(...extendedItems); next = toVisit.shift(); } const items: ApiItem[] = []; for (const members of membersByName.values()) { items.push(...members); } for (const members of membersByKind.values()) { items.push(...members); } items.sort((x: ApiItem, y: ApiItem) => x.getSortKey().localeCompare(y.getSortKey())); return { items, messages, maybeIncompleteResult, }; } /** * @internal */ public _getMergedSiblingsForMember(memberApiItem: ApiItem): readonly ApiItem[] { this._ensureMemberMaps(); let result: ApiItem[] | undefined; if (ApiNameMixin.isBaseClassOf(memberApiItem)) { result = this[_membersByName]!.get(memberApiItem.name); } else { result = this[_membersByKind]!.get(memberApiItem.kind); } if (!result) { throw new InternalError('Item was not found in the _membersByName/_membersByKind lookup'); } return result; } /** * @internal */ public _ensureMemberMaps(): void { // Build the _membersByName and _membersByKind tables if they don't already exist if (this[_membersByName] === undefined) { const membersByName: Map = new Map(); const membersByKind: Map = new Map(); for (const member of this[_members]) { let map: Map | Map; let key: ApiItemKind | string; if (ApiNameMixin.isBaseClassOf(member)) { map = membersByName; key = member.name; } else { map = membersByKind; key = member.kind; } let list: ApiItem[] | undefined = map.get(key); if (list === undefined) { list = []; map.set(key, list); } list.push(member); } this[_membersByName] = membersByName; this[_membersByKind] = membersByKind; } } /** * @override */ public override serializeInto(jsonObject: Partial): void { super.serializeInto(jsonObject); const memberObjects: IApiItemJson[] = []; for (const member of this.members) { const memberJsonObject: Partial = {}; member.serializeInto(memberJsonObject); memberObjects.push(memberJsonObject as IApiItemJson); } jsonObject.preserveMemberOrder = this.preserveMemberOrder; jsonObject.members = memberObjects; } } return MixedClass; } /** * Static members for {@link (ApiItemContainerMixin:interface)}. * * @public */ export namespace ApiItemContainerMixin { /** * A type guard that tests whether the specified `ApiItem` subclass extends the `ApiItemContainerMixin` mixin. * * @remarks * * The JavaScript `instanceof` operator cannot be used to test for mixin inheritance, because each invocation of * the mixin function produces a different subclass. (This could be mitigated by `Symbol.hasInstance`, however * the TypeScript type system cannot invoke a runtime test.) */ export function isBaseClassOf(apiItem: ApiItem): apiItem is ApiItemContainerMixin { return apiItem.hasOwnProperty(_members); } }