mirror of
https://github.com/discordjs/discord.js.git
synced 2026-03-10 00:23:30 +01:00
* fix(ExceptText): don't display import("d..-types/v10"). in return type
* Squashed 'packages/api-extractor-model/' content from commit 39ecb196c
git-subtree-dir: packages/api-extractor-model
git-subtree-split: 39ecb196ca210bdf84ba6c9cadb1bb93571849d7
* Squashed 'packages/api-extractor/' content from commit 341ad6c51
git-subtree-dir: packages/api-extractor
git-subtree-split: 341ad6c51b01656d4f73b74ad4bdb3095f9262c4
* feat(api-extractor): add api-extractor and -model
* fix: package.json docs script
* fix(SourcLink): use <> instead of function syntax
* fix: make packages private
* fix: rest params showing in docs, added labels
* fix: missed two files
* feat: merge docs.json from docgen and docs.api.json
* fix: cpy-cli & pnpm-lock
* fix: increase icon size
* fix: icon size again
* feat: run both docs on mainlib
* chore: website fixes
* fix: more website fixes
* fix: tests and dev database script
* chore: comment out old docs
* fix: increase max fetch cache
* fix: env should always be a string
* fix: try to reapply patches
* fix: remove prepare for docgen
* fix: temporary cosmetic fixes
* fix: horizontal scroll
* feat: generate index for new docs
---------
Co-authored-by: Noel <buechler.noel@outlook.com>
202 lines
7.4 KiB
TypeScript
202 lines
7.4 KiB
TypeScript
// Copyright (c) Microsoft Corporation. All rights reserved. Licensed under the MIT license.
|
|
// See LICENSE in the project root for license information.
|
|
|
|
import * as path from 'node:path';
|
|
import {
|
|
type PackageJsonLookup,
|
|
FileSystem,
|
|
JsonFile,
|
|
type NewlineKind,
|
|
type INodePackageJson,
|
|
type JsonObject,
|
|
} from '@rushstack/node-core-library';
|
|
import { ConsoleMessageId } from '../api/ConsoleMessageId.js';
|
|
import { Extractor } from '../api/Extractor.js';
|
|
import type { MessageRouter } from '../collector/MessageRouter.js';
|
|
|
|
/**
|
|
* Represents analyzed information for a package.json file.
|
|
* This object is constructed and returned by PackageMetadataManager.
|
|
*/
|
|
export class PackageMetadata {
|
|
/**
|
|
* The absolute path to the package.json file being analyzed.
|
|
*/
|
|
public readonly packageJsonPath: string;
|
|
|
|
/**
|
|
* The parsed contents of package.json. Note that PackageJsonLookup
|
|
* only includes essential fields.
|
|
*/
|
|
public readonly packageJson: INodePackageJson;
|
|
|
|
/**
|
|
* If true, then the package's documentation comments can be assumed
|
|
* to contain API Extractor compatible TSDoc tags.
|
|
*/
|
|
public readonly aedocSupported: boolean;
|
|
|
|
public constructor(packageJsonPath: string, packageJson: INodePackageJson, aedocSupported: boolean) {
|
|
this.packageJsonPath = packageJsonPath;
|
|
this.packageJson = packageJson;
|
|
this.aedocSupported = aedocSupported;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* This class maintains a cache of analyzed information obtained from package.json
|
|
* files. It is built on top of the PackageJsonLookup class.
|
|
*
|
|
* @remarks
|
|
*
|
|
* IMPORTANT: Don't use PackageMetadataManager to analyze source files from the current project:
|
|
* 1. Files such as tsdoc-metadata.json may not have been built yet, and thus may contain incorrect information.
|
|
* 2. The current project is not guaranteed to have a package.json file at all. For example, API Extractor can
|
|
* be invoked on a bare .d.ts file.
|
|
*
|
|
* Use ts.program.isSourceFileFromExternalLibrary() to test source files before passing the to PackageMetadataManager.
|
|
*/
|
|
export class PackageMetadataManager {
|
|
public static tsdocMetadataFilename: string = 'tsdoc-metadata.json';
|
|
|
|
private readonly _packageJsonLookup: PackageJsonLookup;
|
|
|
|
private readonly _messageRouter: MessageRouter;
|
|
|
|
private readonly _packageMetadataByPackageJsonPath: Map<string, PackageMetadata> = new Map<string, PackageMetadata>();
|
|
|
|
public constructor(packageJsonLookup: PackageJsonLookup, messageRouter: MessageRouter) {
|
|
this._packageJsonLookup = packageJsonLookup;
|
|
this._messageRouter = messageRouter;
|
|
}
|
|
|
|
// This feature is still being standardized: https://github.com/microsoft/tsdoc/issues/7
|
|
// In the future we will use the @microsoft/tsdoc library to read this file.
|
|
private static _resolveTsdocMetadataPathFromPackageJson(
|
|
packageFolder: string,
|
|
packageJson: INodePackageJson,
|
|
): string {
|
|
const tsdocMetadataFilename: string = PackageMetadataManager.tsdocMetadataFilename;
|
|
|
|
let tsdocMetadataRelativePath: string;
|
|
|
|
if (packageJson.tsdocMetadata) {
|
|
// 1. If package.json contains a field such as "tsdocMetadata": "./path1/path2/tsdoc-metadata.json",
|
|
// then that takes precedence. This convention will be rarely needed, since the other rules below generally
|
|
// produce a good result.
|
|
tsdocMetadataRelativePath = packageJson.tsdocMetadata;
|
|
} else if (packageJson.typings) {
|
|
// 2. If package.json contains a field such as "typings": "./path1/path2/index.d.ts", then we look
|
|
// for the file under "./path1/path2/tsdoc-metadata.json"
|
|
tsdocMetadataRelativePath = path.join(path.dirname(packageJson.typings), tsdocMetadataFilename);
|
|
} else if (packageJson.main) {
|
|
// 3. If package.json contains a field such as "main": "./path1/path2/index.js", then we look for
|
|
// the file under "./path1/path2/tsdoc-metadata.json"
|
|
tsdocMetadataRelativePath = path.join(path.dirname(packageJson.main), tsdocMetadataFilename);
|
|
} else {
|
|
// 4. If none of the above rules apply, then by default we look for the file under "./tsdoc-metadata.json"
|
|
// since the default entry point is "./index.js"
|
|
tsdocMetadataRelativePath = tsdocMetadataFilename;
|
|
}
|
|
|
|
// Always resolve relative to the package folder.
|
|
const tsdocMetadataPath: string = path.resolve(packageFolder, tsdocMetadataRelativePath);
|
|
return tsdocMetadataPath;
|
|
}
|
|
|
|
/**
|
|
* @param packageFolder - The package folder
|
|
* @param packageJson - The package JSON
|
|
* @param tsdocMetadataPath - An explicit path that can be configured in api-extractor.json.
|
|
* If this parameter is not an empty string, it overrides the normal path calculation.
|
|
* @returns the absolute path to the TSDoc metadata file
|
|
*/
|
|
public static resolveTsdocMetadataPath(
|
|
packageFolder: string,
|
|
packageJson: INodePackageJson,
|
|
tsdocMetadataPath?: string,
|
|
): string {
|
|
if (tsdocMetadataPath) {
|
|
return path.resolve(packageFolder, tsdocMetadataPath);
|
|
}
|
|
|
|
return PackageMetadataManager._resolveTsdocMetadataPathFromPackageJson(packageFolder, packageJson);
|
|
}
|
|
|
|
/**
|
|
* Writes the TSDoc metadata file to the specified output file.
|
|
*/
|
|
public static writeTsdocMetadataFile(tsdocMetadataPath: string, newlineKind: NewlineKind): void {
|
|
const fileObject: JsonObject = {
|
|
tsdocVersion: '0.12',
|
|
toolPackages: [
|
|
{
|
|
packageName: '@discordjs/api-extractor',
|
|
packageVersion: Extractor.version,
|
|
},
|
|
],
|
|
};
|
|
|
|
const fileContent: string =
|
|
'// This file is read by tools that parse documentation comments conforming to the TSDoc standard.\n' +
|
|
'// It should be published with your NPM package. It should not be tracked by Git.\n' +
|
|
JsonFile.stringify(fileObject);
|
|
|
|
FileSystem.writeFile(tsdocMetadataPath, fileContent, {
|
|
convertLineEndings: newlineKind,
|
|
ensureFolderExists: true,
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Finds the package.json in a parent folder of the specified source file, and
|
|
* returns a PackageMetadata object. If no package.json was found, then undefined
|
|
* is returned. The results are cached.
|
|
*/
|
|
public tryFetchPackageMetadata(sourceFilePath: string): PackageMetadata | undefined {
|
|
const packageJsonFilePath: string | undefined =
|
|
this._packageJsonLookup.tryGetPackageJsonFilePathFor(sourceFilePath);
|
|
if (!packageJsonFilePath) {
|
|
return undefined;
|
|
}
|
|
|
|
let packageMetadata: PackageMetadata | undefined = this._packageMetadataByPackageJsonPath.get(packageJsonFilePath);
|
|
|
|
if (!packageMetadata) {
|
|
const packageJson: INodePackageJson = this._packageJsonLookup.loadNodePackageJson(packageJsonFilePath);
|
|
|
|
const packageJsonFolder: string = path.dirname(packageJsonFilePath);
|
|
|
|
let aedocSupported = false;
|
|
|
|
const tsdocMetadataPath: string = PackageMetadataManager._resolveTsdocMetadataPathFromPackageJson(
|
|
packageJsonFolder,
|
|
packageJson,
|
|
);
|
|
|
|
if (FileSystem.exists(tsdocMetadataPath)) {
|
|
this._messageRouter.logVerbose(ConsoleMessageId.FoundTSDocMetadata, 'Found metadata in ' + tsdocMetadataPath);
|
|
// If the file exists at all, assume it was written by API Extractor
|
|
aedocSupported = true;
|
|
}
|
|
|
|
packageMetadata = new PackageMetadata(packageJsonFilePath, packageJson, aedocSupported);
|
|
this._packageMetadataByPackageJsonPath.set(packageJsonFilePath, packageMetadata);
|
|
}
|
|
|
|
return packageMetadata;
|
|
}
|
|
|
|
/**
|
|
* Returns true if the source file is part of a package whose .d.ts files support AEDoc annotations.
|
|
*/
|
|
public isAedocSupportedFor(sourceFilePath: string): boolean {
|
|
const packageMetadata: PackageMetadata | undefined = this.tryFetchPackageMetadata(sourceFilePath);
|
|
if (!packageMetadata) {
|
|
return false;
|
|
}
|
|
|
|
return packageMetadata.aedocSupported;
|
|
}
|
|
}
|