D7net
Home
Console
Upload
information
Create File
Create Folder
About
Tools
:
/
home
/
forge
/
ebrokers.online
/
node_modules
/
@tanstack
/
query-core
/
src
/
Filename :
mutation.ts
back
Copy
import { notifyManager } from './notifyManager' import { Removable } from './removable' import { createRetryer } from './retryer' import type { DefaultError, MutationFunctionContext, MutationMeta, MutationOptions, MutationStatus, } from './types' import type { MutationCache } from './mutationCache' import type { MutationObserver } from './mutationObserver' import type { Retryer } from './retryer' import type { QueryClient } from './queryClient' // TYPES interface MutationConfig<TData, TError, TVariables, TOnMutateResult> { client: QueryClient mutationId: number mutationCache: MutationCache options: MutationOptions<TData, TError, TVariables, TOnMutateResult> state?: MutationState<TData, TError, TVariables, TOnMutateResult> } export interface MutationState< TData = unknown, TError = DefaultError, TVariables = unknown, TOnMutateResult = unknown, > { context: TOnMutateResult | undefined data: TData | undefined error: TError | null failureCount: number failureReason: TError | null isPaused: boolean status: MutationStatus variables: TVariables | undefined submittedAt: number } interface FailedAction<TError> { type: 'failed' failureCount: number error: TError | null } interface PendingAction<TVariables, TOnMutateResult> { type: 'pending' isPaused: boolean variables?: TVariables context?: TOnMutateResult } interface SuccessAction<TData> { type: 'success' data: TData } interface ErrorAction<TError> { type: 'error' error: TError } interface PauseAction { type: 'pause' } interface ContinueAction { type: 'continue' } export type Action<TData, TError, TVariables, TOnMutateResult> = | ContinueAction | ErrorAction<TError> | FailedAction<TError> | PendingAction<TVariables, TOnMutateResult> | PauseAction | SuccessAction<TData> // CLASS export class Mutation< TData = unknown, TError = DefaultError, TVariables = unknown, TOnMutateResult = unknown, > extends Removable { state: MutationState<TData, TError, TVariables, TOnMutateResult> options!: MutationOptions<TData, TError, TVariables, TOnMutateResult> readonly mutationId: number #client: QueryClient #observers: Array< MutationObserver<TData, TError, TVariables, TOnMutateResult> > #mutationCache: MutationCache #retryer?: Retryer<TData> constructor( config: MutationConfig<TData, TError, TVariables, TOnMutateResult>, ) { super() this.#client = config.client this.mutationId = config.mutationId this.#mutationCache = config.mutationCache this.#observers = [] this.state = config.state || getDefaultState() this.setOptions(config.options) this.scheduleGc() } setOptions( options: MutationOptions<TData, TError, TVariables, TOnMutateResult>, ): void { this.options = options this.updateGcTime(this.options.gcTime) } get meta(): MutationMeta | undefined { return this.options.meta } addObserver(observer: MutationObserver<any, any, any, any>): void { if (!this.#observers.includes(observer)) { this.#observers.push(observer) // Stop the mutation from being garbage collected this.clearGcTimeout() this.#mutationCache.notify({ type: 'observerAdded', mutation: this, observer, }) } } removeObserver(observer: MutationObserver<any, any, any, any>): void { this.#observers = this.#observers.filter((x) => x !== observer) this.scheduleGc() this.#mutationCache.notify({ type: 'observerRemoved', mutation: this, observer, }) } protected optionalRemove() { if (!this.#observers.length) { if (this.state.status === 'pending') { this.scheduleGc() } else { this.#mutationCache.remove(this) } } } continue(): Promise<unknown> { return ( this.#retryer?.continue() ?? // continuing a mutation assumes that variables are set, mutation must have been dehydrated before this.execute(this.state.variables!) ) } async execute(variables: TVariables): Promise<TData> { const onContinue = () => { this.#dispatch({ type: 'continue' }) } const mutationFnContext = { client: this.#client, meta: this.options.meta, mutationKey: this.options.mutationKey, } satisfies MutationFunctionContext this.#retryer = createRetryer({ fn: () => { if (!this.options.mutationFn) { return Promise.reject(new Error('No mutationFn found')) } return this.options.mutationFn(variables, mutationFnContext) }, onFail: (failureCount, error) => { this.#dispatch({ type: 'failed', failureCount, error }) }, onPause: () => { this.#dispatch({ type: 'pause' }) }, onContinue, retry: this.options.retry ?? 0, retryDelay: this.options.retryDelay, networkMode: this.options.networkMode, canRun: () => this.#mutationCache.canRun(this), }) const restored = this.state.status === 'pending' const isPaused = !this.#retryer.canStart() try { if (restored) { // Dispatch continue action to unpause restored mutation onContinue() } else { this.#dispatch({ type: 'pending', variables, isPaused }) // Notify cache callback await this.#mutationCache.config.onMutate?.( variables, this as Mutation<unknown, unknown, unknown, unknown>, mutationFnContext, ) const context = await this.options.onMutate?.( variables, mutationFnContext, ) if (context !== this.state.context) { this.#dispatch({ type: 'pending', context, variables, isPaused, }) } } const data = await this.#retryer.start() // Notify cache callback await this.#mutationCache.config.onSuccess?.( data, variables, this.state.context, this as Mutation<unknown, unknown, unknown, unknown>, mutationFnContext, ) await this.options.onSuccess?.( data, variables, this.state.context!, mutationFnContext, ) // Notify cache callback await this.#mutationCache.config.onSettled?.( data, null, this.state.variables, this.state.context, this as Mutation<unknown, unknown, unknown, unknown>, mutationFnContext, ) await this.options.onSettled?.( data, null, variables, this.state.context, mutationFnContext, ) this.#dispatch({ type: 'success', data }) return data } catch (error) { try { // Notify cache callback await this.#mutationCache.config.onError?.( error as any, variables, this.state.context, this as Mutation<unknown, unknown, unknown, unknown>, mutationFnContext, ) await this.options.onError?.( error as TError, variables, this.state.context, mutationFnContext, ) // Notify cache callback await this.#mutationCache.config.onSettled?.( undefined, error as any, this.state.variables, this.state.context, this as Mutation<unknown, unknown, unknown, unknown>, mutationFnContext, ) await this.options.onSettled?.( undefined, error as TError, variables, this.state.context, mutationFnContext, ) throw error } finally { this.#dispatch({ type: 'error', error: error as TError }) } } finally { this.#mutationCache.runNext(this) } } #dispatch(action: Action<TData, TError, TVariables, TOnMutateResult>): void { const reducer = ( state: MutationState<TData, TError, TVariables, TOnMutateResult>, ): MutationState<TData, TError, TVariables, TOnMutateResult> => { switch (action.type) { case 'failed': return { ...state, failureCount: action.failureCount, failureReason: action.error, } case 'pause': return { ...state, isPaused: true, } case 'continue': return { ...state, isPaused: false, } case 'pending': return { ...state, context: action.context, data: undefined, failureCount: 0, failureReason: null, error: null, isPaused: action.isPaused, status: 'pending', variables: action.variables, submittedAt: Date.now(), } case 'success': return { ...state, data: action.data, failureCount: 0, failureReason: null, error: null, status: 'success', isPaused: false, } case 'error': return { ...state, data: undefined, error: action.error, failureCount: state.failureCount + 1, failureReason: action.error, isPaused: false, status: 'error', } } } this.state = reducer(this.state) notifyManager.batch(() => { this.#observers.forEach((observer) => { observer.onMutationUpdate(action) }) this.#mutationCache.notify({ mutation: this, type: 'updated', action, }) }) } } export function getDefaultState< TData, TError, TVariables, TOnMutateResult, >(): MutationState<TData, TError, TVariables, TOnMutateResult> { return { context: undefined, data: undefined, error: null, failureCount: 0, failureReason: null, isPaused: false, status: 'idle', variables: undefined, submittedAt: 0, } }