Skip to content

Commit f6078cd

Browse files
committed
feat(core): create scope
To replace `Transient` and `Singleton` patterns.
1 parent 44e2fa9 commit f6078cd

File tree

6 files changed

+140
-4
lines changed

6 files changed

+140
-4
lines changed

src/core/index.ts

+1
Original file line numberDiff line numberDiff line change
@@ -3,3 +3,4 @@ export * from './ikari'
33
export * from './types'
44
export * from './decorators'
55
export * from './utils'
6+
export * from './scope'

src/core/scope/__test__/scope.spec.ts

+55
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,55 @@
1+
import { getInstanceWithScope, TransientScope } from '../'
2+
import { createNewInstance, createOrGetInstanceInScope } from '../utils'
3+
4+
describe('Scope spec:', () => {
5+
describe('createNewInstance', () => {
6+
it('should always return new instance', () => {
7+
class Test {}
8+
9+
expect(createNewInstance(Test)).toBeInstanceOf(Test)
10+
expect(createNewInstance(Test) === createNewInstance(Test)).toBeFalsy()
11+
})
12+
})
13+
14+
describe('createOrGetInstanceInScope', () => {
15+
class Test {}
16+
const scope = 'Scope'
17+
18+
it('should create a new instance if not in scope', () => {
19+
expect(createOrGetInstanceInScope(Test, scope)).toBeInstanceOf(Test)
20+
})
21+
22+
it('should return a same instance if in scope', () => {
23+
expect(createOrGetInstanceInScope(Test, scope)).toBe(createOrGetInstanceInScope(Test, scope))
24+
})
25+
})
26+
27+
describe('getInstanceWithScope', () => {
28+
it('should accept any valid variable as scope', () => {
29+
const scopes = ['', 0, Symbol('symbol'), {}, null]
30+
31+
scopes.forEach((scope) => {
32+
class Test {}
33+
expect(getInstanceWithScope(Test, { scope })).toBeInstanceOf(Test)
34+
})
35+
})
36+
37+
it('should always return same instance if same scope', () => {
38+
const scope = 'scope'
39+
class Test {}
40+
41+
expect(getInstanceWithScope(Test, { scope })).toBe(getInstanceWithScope(Test, { scope }))
42+
})
43+
44+
it('should always return new instance if scope is TransientScope', () => {
45+
class Test {}
46+
47+
expect(getInstanceWithScope(Test, { scope: TransientScope })).toBeInstanceOf(Test)
48+
49+
expect(
50+
getInstanceWithScope(Test, { scope: TransientScope }) ===
51+
getInstanceWithScope(Test, { scope: TransientScope }),
52+
).toBeFalsy()
53+
})
54+
})
55+
})

src/core/scope/index.ts

+25
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
import { InjectableFactory } from '@asuka/di'
2+
import { get } from 'lodash'
3+
4+
import { ConstructorOf } from '../types'
5+
import { ScopeConfig } from './type'
6+
import { createNewInstance, createOrGetInstanceInScope } from './utils'
7+
8+
export { ScopeConfig }
9+
10+
export const TransientScope = Symbol('scope:transient')
11+
12+
export const SingletonScope = Symbol('scope:singleton')
13+
14+
export function getInstanceWithScope<T>(constructor: ConstructorOf<T>, config?: ScopeConfig): T {
15+
const scope = get(config, 'scope', SingletonScope)
16+
17+
switch (scope) {
18+
case SingletonScope:
19+
return InjectableFactory.getInstance(constructor)
20+
case TransientScope:
21+
return createNewInstance(constructor)
22+
default:
23+
return createOrGetInstanceInScope(constructor, scope)
24+
}
25+
}

src/core/scope/type.ts

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
export type Scope = any
2+
3+
export interface ScopeConfig {
4+
scope: Scope
5+
}

src/core/scope/utils.ts

+43
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
import { InjectableFactory } from '@asuka/di'
2+
3+
import { ConstructorOf } from '../types'
4+
import { Scope } from './type'
5+
6+
type ScopeMap<K, V> = Map<K, V>
7+
8+
type Instance = any
9+
10+
type Key = ConstructorOf<Instance>
11+
12+
const map: Map<Key, ScopeMap<Scope, Instance>> = new Map()
13+
14+
export function createNewInstance<T>(constructor: ConstructorOf<T>): T {
15+
return InjectableFactory.injector.resolveAndCreateChild([constructor]).get(constructor)
16+
}
17+
18+
export function createOrGetInstanceInScope<T>(constructor: ConstructorOf<T>, scope: Scope): T {
19+
const instanceAtScope = getInstanceFrom(constructor, scope)
20+
21+
return instanceAtScope ? instanceAtScope : createInstanceInScope(constructor, scope)
22+
}
23+
24+
function createInstanceInScope<T>(constructor: ConstructorOf<T>, scope: Scope): T {
25+
const newInstance = createNewInstance(constructor)
26+
27+
setInstanceInScope(constructor, scope, newInstance)
28+
29+
return newInstance
30+
}
31+
32+
function setInstanceInScope<T>(constructor: ConstructorOf<T>, scope: Scope, newInstance: Instance) {
33+
const scopeMap: ScopeMap<Scope, Instance> = map.get(constructor) || new Map()
34+
35+
scopeMap.set(scope, newInstance)
36+
map.set(constructor, scopeMap)
37+
}
38+
39+
function getInstanceFrom<T>(constructor: ConstructorOf<T>, scope: Scope): T | undefined {
40+
const scopeMap = map.get(constructor)
41+
42+
return scopeMap && scopeMap.get(scope)
43+
}

src/hooks.ts

+11-4
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,13 @@
1-
import { InjectableFactory } from '@asuka/di'
21
import * as React from 'react'
32

4-
import { Ayanami, combineWithIkari, ActionMethodOfAyanami, ConstructorOf } from './core'
3+
import {
4+
Ayanami,
5+
combineWithIkari,
6+
getInstanceWithScope,
7+
ScopeConfig,
8+
ActionMethodOfAyanami,
9+
ConstructorOf,
10+
} from './core'
511

612
export type HooksResult<M extends Ayanami<S>, S> = [Readonly<S>, ActionMethodOfAyanami<M, S>]
713

@@ -18,9 +24,10 @@ export function useAyanamiInstance<M extends Ayanami<S>, S>(ayanami: M): HooksRe
1824
}
1925

2026
export function useAyanami<M extends Ayanami<S>, S>(
21-
constructor: ConstructorOf<M>,
27+
A: ConstructorOf<M>,
28+
config?: ScopeConfig,
2229
): M extends Ayanami<infer SS> ? HooksResult<M, SS> : HooksResult<M, S> {
23-
const ayanami = React.useMemo(() => InjectableFactory.getInstance(constructor), [constructor])
30+
const ayanami = React.useMemo(() => getInstanceWithScope(A, config), [A])
2431

2532
return useAyanamiInstance<M, S>(ayanami) as any
2633
}

0 commit comments

Comments
 (0)