@furystack/shades
Version:
Google Authentication Provider for FuryStack
83 lines (72 loc) • 2.92 kB
text/typescript
import { AggregatedError } from '@furystack/core'
import type { ValueChangeCallback, ValueObserver, ValueObserverOptions } from '@furystack/utils'
import { ObservableValue, isAsyncDisposable, isDisposable } from '@furystack/utils'
/**
* Class for managing observables and disposables for components, based on key-value maps
*/
export class ResourceManager implements AsyncDisposable {
private readonly disposables = new Map<string, Disposable | AsyncDisposable>()
public useDisposable<T extends Disposable | AsyncDisposable>(key: string, factory: () => T): T {
const existing = this.disposables.get(key)
if (!existing) {
const created = factory()
this.disposables.set(key, created)
return created
}
return existing as T
}
public readonly observers = new Map<string, ValueObserver<any>>()
public useObservable = <T>(
key: string,
observable: ObservableValue<T>,
onChange: ValueChangeCallback<T>,
options?: ValueObserverOptions<T>,
): [value: T, setValue: (newValue: T) => void] => {
const alreadyUsed = this.observers.get(key) as ValueObserver<T> | undefined
if (alreadyUsed) {
return [alreadyUsed.observable.getValue(), alreadyUsed.observable.setValue.bind(alreadyUsed.observable)]
}
const observer = observable.subscribe(onChange, options)
this.observers.set(key, observer)
return [observable.getValue(), observable.setValue.bind(observable)]
}
public readonly stateObservers = new Map<string, ObservableValue<any>>()
public useState = <T>(
key: string,
initialValue: T,
callback: ValueChangeCallback<T>,
): [value: T, setValue: (newValue: T) => void] => {
if (!this.stateObservers.has(key)) {
const newObservable = new ObservableValue<T>(initialValue)
this.stateObservers.set(key, newObservable)
newObservable.subscribe(callback)
}
const observable = this.stateObservers.get(key) as ObservableValue<T>
return [observable.getValue(), observable.setValue.bind(observable)]
}
public async [Symbol.asyncDispose]() {
const disposeResult = await Promise.allSettled(
[...this.disposables].map(async ([_key, resource]) => {
if (isDisposable(resource)) {
resource[Symbol.dispose]()
}
if (isAsyncDisposable(resource)) {
await resource[Symbol.asyncDispose]()
}
}),
)
const fails = disposeResult.filter((r) => r.status === 'rejected')
if (fails && fails.length) {
const error = new AggregatedError(
`There was an error during disposing ${fails.length} stores: ${fails.map((f) => f.reason as string).join(', ')}`,
fails,
)
throw error
}
this.disposables.clear()
this.observers.forEach((r) => r[Symbol.dispose]())
this.observers.clear()
this.stateObservers.forEach((r) => r[Symbol.dispose]())
this.stateObservers.clear()
}
}