UNPKG

@nestjs/core

Version:

Nest - modern, fast, powerful node.js web framework (@core)

231 lines (230 loc) 9.83 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.NestContainer = void 0; const constants_1 = require("@nestjs/common/constants"); const discoverable_meta_host_collection_1 = require("../discovery/discoverable-meta-host-collection"); const exceptions_1 = require("../errors/exceptions"); const initialize_on_preview_allowlist_1 = require("../inspector/initialize-on-preview.allowlist"); const serialized_graph_1 = require("../inspector/serialized-graph"); const request_constants_1 = require("../router/request/request-constants"); const compiler_1 = require("./compiler"); const internal_core_module_1 = require("./internal-core-module/internal-core-module"); const internal_providers_storage_1 = require("./internal-providers-storage"); const module_1 = require("./module"); const modules_container_1 = require("./modules-container"); const by_reference_module_opaque_key_factory_1 = require("./opaque-key-factory/by-reference-module-opaque-key-factory"); const deep_hashed_module_opaque_key_factory_1 = require("./opaque-key-factory/deep-hashed-module-opaque-key-factory"); class NestContainer { constructor(_applicationConfig = undefined, _contextOptions = undefined) { this._applicationConfig = _applicationConfig; this._contextOptions = _contextOptions; this.globalModules = new Set(); this.modules = new modules_container_1.ModulesContainer(); this.dynamicModulesMetadata = new Map(); this.internalProvidersStorage = new internal_providers_storage_1.InternalProvidersStorage(); this._serializedGraph = new serialized_graph_1.SerializedGraph(); const moduleOpaqueKeyFactory = this._contextOptions?.moduleIdGeneratorAlgorithm === 'deep-hash' ? new deep_hashed_module_opaque_key_factory_1.DeepHashedModuleOpaqueKeyFactory() : new by_reference_module_opaque_key_factory_1.ByReferenceModuleOpaqueKeyFactory({ keyGenerationStrategy: this._contextOptions?.snapshot ? 'shallow' : 'random', }); this.moduleCompiler = new compiler_1.ModuleCompiler(moduleOpaqueKeyFactory); } get serializedGraph() { return this._serializedGraph; } get applicationConfig() { return this._applicationConfig; } setHttpAdapter(httpAdapter) { this.internalProvidersStorage.httpAdapter = httpAdapter; if (!this.internalProvidersStorage.httpAdapterHost) { return; } const host = this.internalProvidersStorage.httpAdapterHost; host.httpAdapter = httpAdapter; } getHttpAdapterRef() { return this.internalProvidersStorage.httpAdapter; } getHttpAdapterHostRef() { return this.internalProvidersStorage.httpAdapterHost; } async addModule(metatype, scope) { // In DependenciesScanner#scanForModules we already check for undefined or invalid modules // We still need to catch the edge-case of `forwardRef(() => undefined)` if (!metatype) { throw new exceptions_1.UndefinedForwardRefException(scope); } const { type, dynamicMetadata, token } = await this.moduleCompiler.compile(metatype); if (this.modules.has(token)) { return { moduleRef: this.modules.get(token), inserted: true, }; } return { moduleRef: await this.setModule({ token, type, dynamicMetadata, }, scope), inserted: true, }; } async replaceModule(metatypeToReplace, newMetatype, scope) { // In DependenciesScanner#scanForModules we already check for undefined or invalid modules // We still need to catch the edge-case of `forwardRef(() => undefined)` if (!metatypeToReplace || !newMetatype) { throw new exceptions_1.UndefinedForwardRefException(scope); } const { token } = await this.moduleCompiler.compile(metatypeToReplace); const { type, dynamicMetadata } = await this.moduleCompiler.compile(newMetatype); return { moduleRef: await this.setModule({ token, type, dynamicMetadata, }, scope), inserted: false, }; } async setModule({ token, dynamicMetadata, type }, scope) { const moduleRef = new module_1.Module(type, this); moduleRef.token = token; moduleRef.initOnPreview = this.shouldInitOnPreview(type); this.modules.set(token, moduleRef); const updatedScope = [].concat(scope, type); await this.addDynamicMetadata(token, dynamicMetadata, updatedScope); if (this.isGlobalModule(type, dynamicMetadata)) { moduleRef.isGlobal = true; // Set global module distance to MAX_VALUE to ensure their lifecycle hooks // are always executed first (when initializing the application) moduleRef.distance = Number.MAX_VALUE; this.addGlobalModule(moduleRef); } return moduleRef; } async addDynamicMetadata(token, dynamicModuleMetadata, scope) { if (!dynamicModuleMetadata) { return; } this.dynamicModulesMetadata.set(token, dynamicModuleMetadata); const { imports } = dynamicModuleMetadata; await this.addDynamicModules(imports, scope); } async addDynamicModules(modules, scope) { if (!modules) { return; } await Promise.all(modules.map(module => this.addModule(module, scope))); } isGlobalModule(metatype, dynamicMetadata) { if (dynamicMetadata && dynamicMetadata.global) { return true; } return !!Reflect.getMetadata(constants_1.GLOBAL_MODULE_METADATA, metatype); } addGlobalModule(module) { this.globalModules.add(module); } getModules() { return this.modules; } getModuleCompiler() { return this.moduleCompiler; } getModuleByKey(moduleKey) { return this.modules.get(moduleKey); } getInternalCoreModuleRef() { return this.internalCoreModule; } async addImport(relatedModule, token) { if (!this.modules.has(token)) { return; } const moduleRef = this.modules.get(token); const { token: relatedModuleToken } = await this.moduleCompiler.compile(relatedModule); const related = this.modules.get(relatedModuleToken); moduleRef.addImport(related); } addProvider(provider, token, enhancerSubtype) { const moduleRef = this.modules.get(token); if (!provider) { throw new exceptions_1.CircularDependencyException(moduleRef?.metatype.name); } if (!moduleRef) { throw new exceptions_1.UnknownModuleException(); } const providerKey = moduleRef.addProvider(provider, enhancerSubtype); const providerRef = moduleRef.getProviderByKey(providerKey); discoverable_meta_host_collection_1.DiscoverableMetaHostCollection.inspectProvider(this.modules, providerRef); return providerKey; } addInjectable(injectable, token, enhancerSubtype, host) { if (!this.modules.has(token)) { throw new exceptions_1.UnknownModuleException(); } const moduleRef = this.modules.get(token); return moduleRef.addInjectable(injectable, enhancerSubtype, host); } addExportedProviderOrModule(toExport, token) { if (!this.modules.has(token)) { throw new exceptions_1.UnknownModuleException(); } const moduleRef = this.modules.get(token); moduleRef.addExportedProviderOrModule(toExport); } addController(controller, token) { if (!this.modules.has(token)) { throw new exceptions_1.UnknownModuleException(); } const moduleRef = this.modules.get(token); moduleRef.addController(controller); const controllerRef = moduleRef.controllers.get(controller); discoverable_meta_host_collection_1.DiscoverableMetaHostCollection.inspectController(this.modules, controllerRef); } clear() { this.modules.clear(); } replace(toReplace, options) { this.modules.forEach(moduleRef => moduleRef.replace(toReplace, options)); } bindGlobalScope() { this.modules.forEach(moduleRef => this.bindGlobalsToImports(moduleRef)); } bindGlobalsToImports(moduleRef) { this.globalModules.forEach(globalModule => this.bindGlobalModuleToModule(moduleRef, globalModule)); } bindGlobalModuleToModule(target, globalModule) { if (target === globalModule || target === this.internalCoreModule) { return; } target.addImport(globalModule); } getDynamicMetadataByToken(token, metadataKey) { const metadata = this.dynamicModulesMetadata.get(token); return metadataKey ? (metadata?.[metadataKey] ?? []) : metadata; } registerCoreModuleRef(moduleRef) { this.internalCoreModule = moduleRef; this.modules[internal_core_module_1.InternalCoreModule.name] = moduleRef; } getModuleTokenFactory() { return this.moduleCompiler.moduleOpaqueKeyFactory; } registerRequestProvider(request, contextId) { const wrapper = this.internalCoreModule.getProviderByKey(request_constants_1.REQUEST); wrapper.setInstanceByContextId(contextId, { instance: request, isResolved: true, }); } shouldInitOnPreview(type) { return initialize_on_preview_allowlist_1.InitializeOnPreviewAllowlist.has(type); } } exports.NestContainer = NestContainer;