-
Notifications
You must be signed in to change notification settings - Fork 681
Expand file tree
/
Copy pathPluginLoader.ts
More file actions
116 lines (97 loc) · 4.34 KB
/
PluginLoader.ts
File metadata and controls
116 lines (97 loc) · 4.34 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
// 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 * as resolve from 'resolve';
import type { IApiDocumenterPluginManifest, IFeatureDefinition } from './IApiDocumenterPluginManifest.ts';
import {
MarkdownDocumenterFeature,
type MarkdownDocumenterFeatureContext
} from './MarkdownDocumenterFeature.ts';
import { PluginFeatureInitialization } from './PluginFeature.ts';
import type { DocumenterConfig } from '../documenters/DocumenterConfig.ts';
interface ILoadedPlugin {
packageName: string;
manifest: IApiDocumenterPluginManifest;
}
export class PluginLoader {
public markdownDocumenterFeature: MarkdownDocumenterFeature | undefined;
public load(
documenterConfig: DocumenterConfig,
createContext: () => MarkdownDocumenterFeatureContext
): void {
const configFileFolder: string = path.dirname(documenterConfig.configFilePath);
for (const configPlugin of documenterConfig.configFile.plugins || []) {
try {
// Look for the package name in the same place as the config file
const resolvedEntryPointPath: string = resolve.sync(configPlugin.packageName, {
basedir: configFileFolder
});
// Load the package
const entryPoint:
| { apiDocumenterPluginManifest?: IApiDocumenterPluginManifest }
| undefined = require(resolvedEntryPointPath);
if (!entryPoint) {
throw new Error('Invalid entry point');
}
if (!entryPoint.apiDocumenterPluginManifest) {
throw new Error(
`The package is not an API documenter plugin;` +
` the "apiDocumenterPluginManifest" export was not found`
);
}
const manifest: IApiDocumenterPluginManifest = entryPoint.apiDocumenterPluginManifest;
if (manifest.manifestVersion !== 1000) {
throw new Error(
`The plugin is not compatible with this version of API Documenter;` +
` unsupported manifestVersion`
);
}
const loadedPlugin: ILoadedPlugin = {
packageName: configPlugin.packageName,
manifest
};
const featureDefinitionsByName: Map<string, IFeatureDefinition> = new Map<
string,
IFeatureDefinition
>();
for (const featureDefinition of manifest.features) {
featureDefinitionsByName.set(featureDefinition.featureName, featureDefinition);
}
for (const featureName of configPlugin.enabledFeatureNames) {
const featureDefinition: IFeatureDefinition | undefined = featureDefinitionsByName.get(featureName);
if (!featureDefinition) {
throw new Error(
`The plugin ${loadedPlugin.packageName} does not have a feature with name "${featureName}"`
);
}
if (featureDefinition.kind === 'MarkdownDocumenterFeature') {
if (this.markdownDocumenterFeature) {
throw new Error('A MarkdownDocumenterFeature is already loaded');
}
const initialization: PluginFeatureInitialization = new PluginFeatureInitialization();
initialization._context = createContext();
let markdownDocumenterFeature: MarkdownDocumenterFeature | undefined = undefined;
try {
markdownDocumenterFeature = new featureDefinition.subclass(initialization);
} catch (e) {
throw new Error(`Failed to construct feature subclass:\n` + (e as Error).toString());
}
if (!(markdownDocumenterFeature instanceof MarkdownDocumenterFeature)) {
throw new Error('The constructed subclass was not an instance of MarkdownDocumenterFeature');
}
try {
markdownDocumenterFeature.onInitialized();
} catch (e) {
throw new Error('Error occurred during the onInitialized() event: ' + (e as Error).toString());
}
this.markdownDocumenterFeature = markdownDocumenterFeature;
} else {
throw new Error(`Unknown feature definition kind: "${featureDefinition.kind}"`);
}
}
} catch (e) {
throw new Error(`Error loading plugin ${configPlugin.packageName}: ` + (e as Error).message);
}
}
}
}