From 7770a9b5211d7208cfb2bfa5f737d46dc90b7946 Mon Sep 17 00:00:00 2001 From: undergroundwires Date: Thu, 9 Nov 2023 13:17:38 +0100 Subject: [PATCH] Refactor DI for simplicity and type safety This commit improves the dependency injection mechanism by introducing a custom `injectKey` function. Key improvements are: - Enforced type consistency during dependency registration and instantiation. - Simplified injection process, abstracting away the complexity with a uniform API, regardless of the dependency's lifetime. - Eliminated the possibility of `undefined` returns during dependency injection, promoting fail-fast behavior. - Removed the necessity for type casting to `symbol` for injection keys in unit tests by using existing types. - Consalidated imports, combining keys and injection functions in one `import` statement. --- docs/presentation.md | 9 +- .../bootstrapping/DependencyProvider.ts | 93 ++++++++++++++----- .../Code/CodeButtons/CodeCopyButton.vue | 10 +- .../Code/CodeButtons/CodeRunButton.vue | 10 +- .../Code/CodeButtons/Save/CodeSaveButton.vue | 8 +- .../Save/Instructions/CodeInstruction.vue | 6 +- .../Save/Instructions/InstructionList.vue | 5 +- .../Code/CodeButtons/TheCodeButtons.vue | 6 +- .../components/Code/TheCodeArea.vue | 8 +- .../Scripts/Menu/Selector/TheSelector.vue | 8 +- .../components/Scripts/Menu/TheOsChanger.vue | 10 +- .../Scripts/Menu/TheScriptsMenu.vue | 10 +- .../Scripts/View/Cards/CardList.vue | 5 +- .../Scripts/View/Cards/CardListItem.vue | 9 +- .../Scripts/View/TheScriptsView.vue | 9 +- .../View/Tree/NodeContent/RevertToggle.vue | 9 +- .../View/Tree/TreeView/Node/UseNodeState.ts | 6 +- .../View/Tree/TreeView/UseCurrentTreeNodes.ts | 6 +- .../TreeView/UseNodeStateChangeAggregator.ts | 6 +- .../UseCollectionSelectionStateUpdater.ts | 5 +- .../UseSelectedScriptNodeIds.ts | 8 +- .../TreeViewAdapter/UseTreeViewFilterEvent.ts | 8 +- .../TreeViewAdapter/UseTreeViewNodeInput.ts | 6 +- .../components/TheFooter/DownloadUrlList.vue | 6 +- .../TheFooter/DownloadUrlListItem.vue | 7 +- .../components/TheFooter/PrivacyPolicy.vue | 8 +- .../components/TheFooter/TheFooter.vue | 8 +- src/presentation/components/TheHeader.vue | 6 +- src/presentation/components/TheSearchBar.vue | 7 +- src/presentation/injectionSymbols.ts | 68 +++++++++++++- .../composite/DependencyResolution.spec.ts | 53 +++++++++++ tests/integration/composite/README.md | 3 + .../ApplicationBootstrapper.spec.ts | 16 ++++ .../shared/Assertions/ExpectThrowsAsync.ts | 0 .../Context/ApplicationContextFactory.spec.ts | 2 +- tests/unit/infrastructure/CodeRunner.spec.ts | 2 +- .../ApplicationBootstrapper.spec.ts | 2 +- .../bootstrapping/DependencyProvider.spec.ts | 5 +- .../Modules/RuntimeSanityValidator.spec.ts | 2 +- .../Code/CodeButtons/CodeCopyButton.spec.ts | 4 +- .../Save/Instructions/CodeInstruction.spec.ts | 4 +- .../Scripts/View/TheScriptsView.spec.ts | 6 +- .../Tree/TreeView/Node/UseNodeState.spec.ts | 2 +- .../Tree/TreeView/UseCurrentTreeNodes.spec.ts | 2 +- .../UseNodeStateChangeAggregator.spec.ts | 2 +- ...UseCollectionSelectionStateUpdater.spec.ts | 2 +- .../UseSelectedScriptNodeIds.spec.ts | 4 +- .../UseTreeViewFilterEvent.spec.ts | 4 +- .../UseTreeViewNodeInput.spec.ts | 2 +- .../Hooks/Clipboard/BrowserClipboard.spec.ts | 2 +- tests/unit/presentation/injectionSymbols.ts | 86 +++++++++++++---- 51 files changed, 398 insertions(+), 177 deletions(-) create mode 100644 tests/integration/composite/DependencyResolution.spec.ts create mode 100644 tests/integration/composite/README.md create mode 100644 tests/integration/presentation/bootstrapping/ApplicationBootstrapper.spec.ts rename tests/{unit => }/shared/Assertions/ExpectThrowsAsync.ts (100%) diff --git a/docs/presentation.md b/docs/presentation.md index 1ace9c2e..d5cdb281 100644 --- a/docs/presentation.md +++ b/docs/presentation.md @@ -71,10 +71,11 @@ To add a new dependency: 1. **Define its symbol**: Define an associated symbol for every dependency in [`injectionSymbols.ts`](./../src/presentation/injectionSymbols.ts). Symbols are grouped into: - **Singletons**: Shared across components, instantiated once. - **Transients**: Factories yielding a new instance on every access. -2. **Provide the dependency**: Modify the [`provideDependencies`](./../src/presentation/bootstrapping/DependencyProvider.ts) function to include the new dependency. [`App.vue`](./../src/presentation/components/App.vue) calls this function within its `setup()` hook to register the dependencies. -3. **Inject the dependency**: Use Vue's `inject` method alongside the defined symbol to incorporate the dependency into components. - - For singletons, invoke the factory method: `inject(symbolKey)()`. - - For transients, directly inject: `inject(symbolKey)`. +2. **Provide the dependency**: + Modify the [`provideDependencies`](./../src/presentation/bootstrapping/DependencyProvider.ts) function to include the new dependency. + [`App.vue`](./../src/presentation/components/App.vue) calls this function within its `setup()` hook to register the dependencies. +3. **Inject the dependency**: Use `injectKey` to inject a dependency. Pass a selector function to `injectKey` that retrieves the appropriate symbol from the provided dependencies. + - Example usage: `injectKey((keys) => keys.useCollectionState)`; ## Shared UI components diff --git a/src/presentation/bootstrapping/DependencyProvider.ts b/src/presentation/bootstrapping/DependencyProvider.ts index dc392c02..09f67d0b 100644 --- a/src/presentation/bootstrapping/DependencyProvider.ts +++ b/src/presentation/bootstrapping/DependencyProvider.ts @@ -2,38 +2,89 @@ import { InjectionKey, provide, inject } from 'vue'; import { useCollectionState } from '@/presentation/components/Shared/Hooks/UseCollectionState'; import { useApplication } from '@/presentation/components/Shared/Hooks/UseApplication'; import { useAutoUnsubscribedEvents } from '@/presentation/components/Shared/Hooks/UseAutoUnsubscribedEvents'; +import { useClipboard } from '@/presentation/components/Shared/Hooks/Clipboard/UseClipboard'; +import { useCurrentCode } from '@/presentation/components/Shared/Hooks/UseCurrentCode'; import { IApplicationContext } from '@/application/Context/IApplicationContext'; import { RuntimeEnvironment } from '@/infrastructure/RuntimeEnvironment/RuntimeEnvironment'; -import { InjectionKeys } from '@/presentation/injectionSymbols'; -import { useClipboard } from '../components/Shared/Hooks/Clipboard/UseClipboard'; -import { useCurrentCode } from '../components/Shared/Hooks/UseCurrentCode'; +import { + AnyLifetimeInjectionKey, InjectionKeySelector, InjectionKeys, SingletonKey, + TransientKey, injectKey, +} from '@/presentation/injectionSymbols'; +import { PropertyKeys } from '@/TypeHelpers'; export function provideDependencies( context: IApplicationContext, api: VueDependencyInjectionApi = { provide, inject }, ) { - const registerSingleton = (key: InjectionKey, value: T) => api.provide(key, value); - const registerTransient = ( - key: InjectionKey<() => T>, - factory: () => T, - ) => api.provide(key, factory); + const resolvers: Record, (di: DependencyRegistrar) => void> = { + useCollectionState: (di) => di.provide( + InjectionKeys.useCollectionState, + () => { + const { events } = di.injectKey((keys) => keys.useAutoUnsubscribedEvents); + return useCollectionState(context, events); + }, + ), + useApplication: (di) => di.provide( + InjectionKeys.useApplication, + useApplication(context.app), + ), + useRuntimeEnvironment: (di) => di.provide( + InjectionKeys.useRuntimeEnvironment, + RuntimeEnvironment.CurrentEnvironment, + ), + useAutoUnsubscribedEvents: (di) => di.provide( + InjectionKeys.useAutoUnsubscribedEvents, + useAutoUnsubscribedEvents, + ), + useClipboard: (di) => di.provide( + InjectionKeys.useClipboard, + useClipboard, + ), + useCurrentCode: (di) => di.provide( + InjectionKeys.useCurrentCode, + () => { + const { events } = di.injectKey((keys) => keys.useAutoUnsubscribedEvents); + const state = di.injectKey((keys) => keys.useCollectionState); + return useCurrentCode(state, events); + }, + ), + }; + registerAll(Object.values(resolvers), api); +} - registerSingleton(InjectionKeys.useApplication, useApplication(context.app)); - registerSingleton(InjectionKeys.useRuntimeEnvironment, RuntimeEnvironment.CurrentEnvironment); - registerTransient(InjectionKeys.useAutoUnsubscribedEvents, () => useAutoUnsubscribedEvents()); - registerTransient(InjectionKeys.useCollectionState, () => { - const { events } = api.inject(InjectionKeys.useAutoUnsubscribedEvents)(); - return useCollectionState(context, events); - }); - registerTransient(InjectionKeys.useClipboard, () => useClipboard()); - registerTransient(InjectionKeys.useCurrentCode, () => { - const { events } = api.inject(InjectionKeys.useAutoUnsubscribedEvents)(); - const state = api.inject(InjectionKeys.useCollectionState)(); - return useCurrentCode(state, events); - }); +function registerAll( + registrations: ReadonlyArray<(di: DependencyRegistrar) => void>, + api: VueDependencyInjectionApi, +) { + const registrar = new DependencyRegistrar(api); + Object.values(registrations).forEach((register) => register(registrar)); } export interface VueDependencyInjectionApi { provide(key: InjectionKey, value: T): void; inject(key: InjectionKey): T; } + +class DependencyRegistrar { + constructor(private api: VueDependencyInjectionApi) {} + + public provide( + key: TransientKey, + resolver: () => T, + ): void; + public provide( + key: SingletonKey, + resolver: T, + ): void; + public provide( + key: AnyLifetimeInjectionKey, + resolver: T | (() => T), + ): void { + this.api.provide(key.key, resolver); + } + + public injectKey(key: InjectionKeySelector): T { + const injector = this.api.inject.bind(this.api); + return injectKey(key, injector); + } +} diff --git a/src/presentation/components/Code/CodeButtons/CodeCopyButton.vue b/src/presentation/components/Code/CodeButtons/CodeCopyButton.vue index ab20237e..3c38cdba 100644 --- a/src/presentation/components/Code/CodeButtons/CodeCopyButton.vue +++ b/src/presentation/components/Code/CodeButtons/CodeCopyButton.vue @@ -7,10 +7,8 @@