马宇豪
2024-07-16 f591c27b57e2418c9495bc02ae8cfff84d35bc18
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
/*
    MIT License http://www.opensource.org/licenses/mit-license.php
    Author Tobias Koppers @sokra
*/
 
"use strict";
 
const path = require("path");
const DescriptionFileUtils = require("./DescriptionFileUtils");
 
/** @typedef {import("./Resolver")} Resolver */
/** @typedef {import("./Resolver").JsonObject} JsonObject */
/** @typedef {import("./Resolver").ResolveRequest} ResolveRequest */
/** @typedef {import("./Resolver").ResolveStepHook} ResolveStepHook */
 
/** @typedef {{name: string|Array<string>, forceRelative: boolean}} MainFieldOptions */
 
const alreadyTriedMainField = Symbol("alreadyTriedMainField");
 
module.exports = class MainFieldPlugin {
    /**
     * @param {string | ResolveStepHook} source source
     * @param {MainFieldOptions} options options
     * @param {string | ResolveStepHook} target target
     */
    constructor(source, options, target) {
        this.source = source;
        this.options = options;
        this.target = target;
    }
 
    /**
     * @param {Resolver} resolver the resolver
     * @returns {void}
     */
    apply(resolver) {
        const target = resolver.ensureHook(this.target);
        resolver
            .getHook(this.source)
            .tapAsync("MainFieldPlugin", (request, resolveContext, callback) => {
                if (
                    request.path !== request.descriptionFileRoot ||
                    /** @type {ResolveRequest & { [alreadyTriedMainField]?: string }} */
                    (request)[alreadyTriedMainField] === request.descriptionFilePath ||
                    !request.descriptionFilePath
                )
                    return callback();
                const filename = path.basename(request.descriptionFilePath);
                let mainModule =
                    /** @type {string|null|undefined} */
                    (
                        DescriptionFileUtils.getField(
                            /** @type {JsonObject} */ (request.descriptionFileData),
                            this.options.name
                        )
                    );
 
                if (
                    !mainModule ||
                    typeof mainModule !== "string" ||
                    mainModule === "." ||
                    mainModule === "./"
                ) {
                    return callback();
                }
                if (this.options.forceRelative && !/^\.\.?\//.test(mainModule))
                    mainModule = "./" + mainModule;
                /** @type {ResolveRequest & { [alreadyTriedMainField]?: string }} */
                const obj = {
                    ...request,
                    request: mainModule,
                    module: false,
                    directory: mainModule.endsWith("/"),
                    [alreadyTriedMainField]: request.descriptionFilePath
                };
                return resolver.doResolve(
                    target,
                    obj,
                    "use " +
                        mainModule +
                        " from " +
                        this.options.name +
                        " in " +
                        filename,
                    resolveContext,
                    callback
                );
            });
    }
};