obsidian-dev-utils
Version:
This is the collection of useful functions that you can use for your Obsidian plugin development
467 lines (454 loc) • 55.3 kB
JavaScript
/*
THIS IS A GENERATED/BUNDLED FILE BY ESBUILD
if you want to view the source, please visit the github repository of this plugin
*/
(function initCjs() {
const globalThisRecord = globalThis;
globalThisRecord['__name'] ??= name;
const originalRequire = require;
if (originalRequire && !originalRequire.__isPatched) {
// eslint-disable-next-line no-global-assign, no-implicit-globals -- We need to patch the `require()` function.
require = Object.assign(
(id) => requirePatched(id),
originalRequire,
{
__isPatched: true
}
);
}
const newFuncs = {
__extractDefault() {
return extractDefault;
},
process() {
const browserProcess = {
browser: true,
cwd() {
return '/';
},
env: {},
platform: 'android'
};
return browserProcess;
}
};
for (const key of Object.keys(newFuncs)) {
globalThisRecord[key] ??= newFuncs[key]?.();
}
function name(obj) {
return obj;
}
function extractDefault(module) {
return module && module.__esModule && 'default' in module ? module.default : module;
}
const OBSIDIAN_BUILT_IN_MODULE_NAMES = [
'obsidian',
'@codemirror/autocomplete',
'@codemirror/collab',
'@codemirror/commands',
'@codemirror/language',
'@codemirror/lint',
'@codemirror/search',
'@codemirror/state',
'@codemirror/text',
'@codemirror/view',
'@lezer/common',
'@lezer/lr',
'@lezer/highlight'];
const DEPRECATED_OBSIDIAN_BUILT_IN_MODULE_NAMES = [
'@codemirror/closebrackets',
'@codemirror/comment',
'@codemirror/fold',
'@codemirror/gutter',
'@codemirror/highlight',
'@codemirror/history',
'@codemirror/matchbrackets',
'@codemirror/panel',
'@codemirror/rangeset',
'@codemirror/rectangular-selection',
'@codemirror/stream-parser',
'@codemirror/tooltip'];
function requirePatched(id) {
if (OBSIDIAN_BUILT_IN_MODULE_NAMES.includes(id) || DEPRECATED_OBSIDIAN_BUILT_IN_MODULE_NAMES.includes(id)) {
return originalRequire?.(id);
}
// eslint-disable-next-line @typescript-eslint/no-deprecated, @typescript-eslint/no-unnecessary-condition -- We need access to app here which might not be available yet.
if (globalThis?.app?.isMobile) {
if (id === 'process' || id === 'node:process') {
console.debug(`The most likely you can safely ignore this error. Module not found: ${id}. Fake process object is returned instead.`);
return globalThis.process;
}
} else {
const module = originalRequire?.(id);
if (module) {
return extractDefault(module);
}
}
console.debug(`The most likely you can safely ignore this error. Module not found: ${id}. Empty object is returned instead.`);
return {};
}
})();
"use strict";
var __defProp = Object.defineProperty;
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
var __getOwnPropNames = Object.getOwnPropertyNames;
var __hasOwnProp = Object.prototype.hasOwnProperty;
var __export = (target, all) => {
for (var name in all)
__defProp(target, name, { get: all[name], enumerable: true });
};
var __copyProps = (to, from, except, desc) => {
if (from && typeof from === "object" || typeof from === "function") {
for (let key of __getOwnPropNames(from))
if (!__hasOwnProp.call(to, key) && key !== except)
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
}
return to;
};
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
var Async_exports = {};
__export(Async_exports, {
addErrorHandler: () => addErrorHandler,
asyncFilter: () => asyncFilter,
asyncFilterInPlace: () => asyncFilterInPlace,
asyncFlatMap: () => asyncFlatMap,
asyncMap: () => asyncMap,
convertAsyncToSync: () => convertAsyncToSync,
convertSyncToAsync: () => convertSyncToAsync,
handleSilentError: () => handleSilentError,
ignoreError: () => ignoreError,
invokeAsyncSafely: () => invokeAsyncSafely,
invokeAsyncSafelyAfterDelay: () => invokeAsyncSafelyAfterDelay,
marksAsTerminateRetry: () => marksAsTerminateRetry,
neverEnds: () => neverEnds,
nextTickAsync: () => nextTickAsync,
promiseAllAsyncFnsSequentially: () => promiseAllAsyncFnsSequentially,
promiseAllSequentially: () => promiseAllSequentially,
queueMicrotaskAsync: () => queueMicrotaskAsync,
requestAnimationFrameAsync: () => requestAnimationFrameAsync,
retryWithTimeout: () => retryWithTimeout,
runWithTimeout: () => runWithTimeout,
setImmediateAsync: () => setImmediateAsync,
setTimeoutAsync: () => setTimeoutAsync,
sleep: () => sleep,
timeout: () => timeout,
toArray: () => toArray
});
module.exports = __toCommonJS(Async_exports);
var import_AbortController = require('./AbortController.cjs');
var import_Debug = require('./Debug.cjs');
var import_Error = require('./Error.cjs');
var import_Function = require('./Function.cjs');
var import_ObjectUtils = require('./ObjectUtils.cjs');
async function addErrorHandler(asyncFn, stackTrace) {
stackTrace ??= (0, import_Error.getStackTrace)(1);
try {
await asyncFn();
} catch (asyncError) {
const wrappedError = new import_Error.CustomStackTraceError(import_Error.ASYNC_WRAPPER_ERROR_MESSAGE, stackTrace, asyncError);
if (handleSilentError(wrappedError)) {
return;
}
(0, import_Error.emitAsyncErrorEvent)(wrappedError);
}
}
async function asyncFilter(arr, predicate) {
const ans = [];
const length = arr.length;
for (let i = 0; i < length; i++) {
if (!Object.hasOwn(arr, i)) {
continue;
}
const item = arr[i];
if (await predicate(item, i, arr)) {
ans.push(item);
}
}
return ans;
}
async function asyncFilterInPlace(arr, predicate) {
const length = arr.length;
let writeIndex = 0;
for (let readIndex = 0; readIndex < length; readIndex++) {
if (!Object.hasOwn(arr, readIndex)) {
continue;
}
const current = arr[readIndex];
if (await predicate(current, readIndex, arr)) {
arr[writeIndex++] = current;
}
}
arr.length = writeIndex;
}
async function asyncFlatMap(arr, callback) {
return (await asyncMap(arr, callback)).flat();
}
async function asyncMap(arr, callback) {
return await promiseAllSequentially(arr.map(callback));
}
function convertAsyncToSync(asyncFunc, stackTrace) {
stackTrace ??= (0, import_Error.getStackTrace)(1);
return (...args) => {
const innerStackTrace = (0, import_Error.getStackTrace)(1);
stackTrace = `${stackTrace ?? ""}
at --- convertAsyncToSync --- (0)
${innerStackTrace}`;
invokeAsyncSafely(() => asyncFunc(...args), stackTrace);
};
}
function convertSyncToAsync(syncFn) {
return async (...args) => {
await Promise.resolve();
return syncFn(...args);
};
}
function handleSilentError(error) {
let cause = error;
while (!(cause instanceof import_Error.SilentError)) {
if (!(cause instanceof Error)) {
return false;
}
cause = cause.cause;
}
(0, import_Debug.getLibDebugger)("Async:handleSilentError")(error);
return true;
}
async function ignoreError(promise, fallbackValue) {
const ignoreErrorDebugger = (0, import_Debug.getLibDebugger)("Async:ignoreError");
const stackTrace = (0, import_Error.getStackTrace)(1);
try {
return await promise;
} catch (e) {
ignoreErrorDebugger("Ignored error", new import_Error.CustomStackTraceError("Ignored error", stackTrace, e));
return fallbackValue;
}
}
function invokeAsyncSafely(asyncFn, stackTrace) {
stackTrace ??= (0, import_Error.getStackTrace)(1);
void addErrorHandler(asyncFn, stackTrace);
}
function invokeAsyncSafelyAfterDelay(asyncFn, delayInMilliseconds = 0, stackTrace, abortSignal) {
abortSignal ??= (0, import_AbortController.abortSignalNever)();
abortSignal.throwIfAborted();
stackTrace ??= (0, import_Error.getStackTrace)(1);
invokeAsyncSafely(async () => {
await sleep(delayInMilliseconds, abortSignal, true);
await asyncFn(abortSignal);
}, stackTrace);
}
async function promiseAllAsyncFnsSequentially(asyncFns) {
const results = [];
for (const asyncFn of asyncFns) {
results.push(await asyncFn());
}
return results;
}
async function promiseAllSequentially(promises) {
return await promiseAllAsyncFnsSequentially(promises.map((promise) => () => promise));
}
const terminateRetryErrors = /* @__PURE__ */ new WeakSet();
function marksAsTerminateRetry(error) {
terminateRetryErrors.add(error);
}
async function neverEnds() {
await new Promise(() => {
(0, import_Function.noop)();
});
throw new Error("Should never happen");
}
async function nextTickAsync() {
return new Promise((resolve) => {
process.nextTick(() => {
resolve();
});
});
}
async function queueMicrotaskAsync() {
return new Promise((resolve) => {
queueMicrotask(() => {
resolve();
});
});
}
async function requestAnimationFrameAsync() {
return new Promise((resolve) => {
requestAnimationFrame(() => {
resolve();
});
});
}
async function retryWithTimeout(options) {
const retryWithTimeoutDebugger = (0, import_Debug.getLibDebugger)("Async:retryWithTimeout");
const stackTrace = options.stackTrace ?? (0, import_Error.getStackTrace)(1);
const DEFAULT_RETRY_OPTIONS = {
// eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.
retryDelayInMilliseconds: 100,
shouldRetryOnError: false,
// eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.
timeoutInMilliseconds: 5e3
};
const fullOptions = { ...DEFAULT_RETRY_OPTIONS, ...options.retryOptions };
fullOptions.abortSignal?.throwIfAborted();
await runWithTimeout((0, import_ObjectUtils.normalizeOptionalProperties)({
context: { operationName: options.operationName ?? "", retryFn: options.operationFn },
onTimeout: options.onTimeout,
async operationFn(abortSignal) {
const combinedAbortSignal = (0, import_AbortController.abortSignalAny)(fullOptions.abortSignal, abortSignal);
combinedAbortSignal.throwIfAborted();
let attempt = 0;
while (!combinedAbortSignal.aborted) {
attempt++;
let isSuccess;
try {
isSuccess = await options.operationFn(combinedAbortSignal);
} catch (error) {
if (combinedAbortSignal.aborted || !fullOptions.shouldRetryOnError || terminateRetryErrors.has(error)) {
throw new import_Error.CustomStackTraceError("retryWithTimeout failed", stackTrace, error);
}
(0, import_Error.printError)(error);
isSuccess = false;
}
if (isSuccess) {
(0, import_Debug.printWithStackTrace)(retryWithTimeoutDebugger, stackTrace, `Retry completed successfully after ${String(attempt)} attempts`, {
operationFn: options.operationFn,
operationName: options.operationName ?? ""
});
return;
}
(0, import_Debug.printWithStackTrace)(
retryWithTimeoutDebugger,
stackTrace,
`Retry attempt ${String(attempt)} completed unsuccessfully. Trying again in ${String(fullOptions.retryDelayInMilliseconds)} milliseconds`,
{
operationFn: options.operationFn,
operationName: options.operationName ?? ""
}
);
await sleep(fullOptions.retryDelayInMilliseconds, abortSignal);
}
},
operationName: options.operationName ?? "",
stackTrace,
timeoutInMilliseconds: fullOptions.timeoutInMilliseconds
}));
}
async function runWithTimeout(options) {
const stackTrace = options.stackTrace ?? (0, import_Error.getStackTrace)(1);
const startTime = performance.now();
const runAbortController = new AbortController();
const timeoutAbortController = new AbortController();
let result = null;
let hasResult = false;
let isCompleted = false;
const runWithTimeoutDebugger = (0, import_Debug.getLibDebugger)("Async:runWithTimeout");
const onTimeout = options.onTimeout ?? defaultOnTimeout;
await Promise.race([run(), innerTimeout()]);
if (hasResult) {
return result;
}
throw new import_Error.CustomStackTraceError("Run with timeout failed", stackTrace, runAbortController.signal.reason);
async function run() {
try {
result = await options.operationFn(runAbortController.signal);
const duration = Math.trunc(performance.now() - startTime);
(0, import_Debug.printWithStackTrace)(runWithTimeoutDebugger, stackTrace, `Execution time: ${String(duration)} milliseconds`, {
context: options.context,
operationFn: options.operationFn,
operationName: options.operationName ?? ""
});
hasResult = true;
} catch (e) {
runAbortController.abort(e);
} finally {
isCompleted = true;
timeoutAbortController.abort(new Error("Completed"));
}
}
async function innerTimeout() {
await sleep(options.timeoutInMilliseconds, timeoutAbortController.signal);
if (isCompleted) {
return;
}
const duration = Math.trunc(performance.now() - startTime);
(0, import_Debug.printWithStackTrace)(runWithTimeoutDebugger, stackTrace, `Timed out after ${String(duration)} milliseconds`, {
context: options.context,
operationFn: options.operationFn,
operationName: options.operationName ?? ""
});
const timeoutContext = (0, import_ObjectUtils.normalizeOptionalProperties)({
duration,
onOperationCompleted(callback) {
timeoutAbortController.signal.addEventListener("abort", callback);
},
operationName: options.operationName ?? "",
terminateOperation() {
const error = new Error(`Timed out after ${String(duration)} milliseconds`);
runAbortController.abort(error);
timeoutAbortController.abort(error);
}
});
onTimeout(timeoutContext);
await (0, import_AbortController.waitForAbort)(timeoutAbortController.signal);
}
function defaultOnTimeout(ctx) {
ctx.terminateOperation();
}
}
async function setImmediateAsync() {
return new Promise((resolve) => {
setImmediate(() => {
resolve();
});
});
}
async function setTimeoutAsync(delay) {
await new Promise((resolve) => {
setTimeout(resolve, delay);
});
}
async function sleep(milliseconds, abortSignal, shouldThrowOnAbort) {
await (0, import_AbortController.waitForAbort)((0, import_AbortController.abortSignalAny)(abortSignal, (0, import_AbortController.abortSignalTimeout)(milliseconds)));
if (shouldThrowOnAbort) {
abortSignal?.throwIfAborted();
}
}
async function timeout(timeoutInMilliseconds, abortSignal, shouldThrowOnAbort) {
await sleep(timeoutInMilliseconds, abortSignal, shouldThrowOnAbort);
throw new Error(`Timed out in ${String(timeoutInMilliseconds)} milliseconds`);
}
async function toArray(iter) {
const arr = [];
for await (const item of iter) {
arr.push(item);
}
return arr;
}
// Annotate the CommonJS export names for ESM import in node:
0 && (module.exports = {
addErrorHandler,
asyncFilter,
asyncFilterInPlace,
asyncFlatMap,
asyncMap,
convertAsyncToSync,
convertSyncToAsync,
handleSilentError,
ignoreError,
invokeAsyncSafely,
invokeAsyncSafelyAfterDelay,
marksAsTerminateRetry,
neverEnds,
nextTickAsync,
promiseAllAsyncFnsSequentially,
promiseAllSequentially,
queueMicrotaskAsync,
requestAnimationFrameAsync,
retryWithTimeout,
runWithTimeout,
setImmediateAsync,
setTimeoutAsync,
sleep,
timeout,
toArray
});
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/Async.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility functions for asynchronous operations.\n */\n\nimport type { Promisable } from 'type-fest';\n\nimport {\n  abortSignalAny,\n  abortSignalNever,\n  abortSignalTimeout,\n  waitForAbort\n} from './AbortController.ts';\nimport {\n  getLibDebugger,\n  printWithStackTrace\n} from './Debug.ts';\nimport {\n  ASYNC_WRAPPER_ERROR_MESSAGE,\n  CustomStackTraceError,\n  emitAsyncErrorEvent,\n  getStackTrace,\n  printError,\n  SilentError\n} from './Error.ts';\nimport { noop } from './Function.ts';\nimport { normalizeOptionalProperties } from './ObjectUtils.ts';\n\n/**\n * A type representing a function that resolves a {@link Promise}.\n *\n * @typeParam T - The type of the value.\n */\nexport type PromiseResolve<T> = undefined extends T ? (value?: PromiseLike<T> | T) => void\n  : (value: PromiseLike<T> | T) => void;\n\n/**\n * Options for {@link retryWithTimeout}.\n */\nexport interface RetryOptions {\n  /**\n   * A abort signal to cancel the retry operation.\n   */\n  abortSignal?: AbortSignal;\n\n  /**\n   * A delay in milliseconds between retry attempts.\n   */\n  retryDelayInMilliseconds?: number;\n\n  /**\n   * Whether to retry the function on error.\n   */\n  shouldRetryOnError?: boolean;\n\n  /**\n   * A maximum time in milliseconds to wait before giving up on retrying.\n   */\n  timeoutInMilliseconds?: number;\n}\n\n/**\n * Adds an error handler to a {@link Promise} that catches any errors and emits an async error event.\n *\n * @param asyncFn - The asynchronous function to add an error handler to.\n * @param stackTrace - The stack trace of the source function.\n * @returns A {@link Promise} that resolves when the asynchronous function completes or emits async error event.\n */\nexport async function addErrorHandler(asyncFn: () => Promise<unknown>, stackTrace?: string): Promise<void> {\n  stackTrace ??= getStackTrace(1);\n  try {\n    await asyncFn();\n  } catch (asyncError) {\n    const wrappedError = new CustomStackTraceError(ASYNC_WRAPPER_ERROR_MESSAGE, stackTrace, asyncError);\n    if (handleSilentError(wrappedError)) {\n      return;\n    }\n    emitAsyncErrorEvent(wrappedError);\n  }\n}\n\n/**\n * Filters an array asynchronously, keeping only the elements that satisfy the provided predicate function.\n *\n * @typeParam T - The type of elements in the input array.\n * @param arr - The array to filter.\n * @param predicate - The predicate function to test each element.\n * @returns A {@link Promise} that resolves with an array of elements that satisfy the predicate function.\n */\nexport async function asyncFilter<T>(arr: T[], predicate: (value: T, index: number, array: T[]) => Promisable<boolean>): Promise<T[]> {\n  const ans: T[] = [];\n\n  const length = arr.length;\n  for (let i = 0; i < length; i++) {\n    if (!Object.hasOwn(arr, i)) {\n      continue;\n    }\n\n    const item = arr[i] as T;\n    if (await predicate(item, i, arr)) {\n      ans.push(item);\n    }\n  }\n\n  return ans;\n}\n\n/**\n * Filters an array asynchronously in place, keeping only the elements that satisfy the provided predicate function.\n *\n * @typeParam T - The type of elements in the input array.\n * @param arr - The array to filter.\n * @param predicate - The predicate function to test each element.\n * @returns A {@link Promise} that resolves when the array is filtered.\n */\nexport async function asyncFilterInPlace<T>(arr: T[], predicate: (value: T, index: number, array: T[]) => Promisable<boolean>): Promise<void> {\n  const length = arr.length;\n  let writeIndex = 0;\n  for (let readIndex = 0; readIndex < length; readIndex++) {\n    if (!Object.hasOwn(arr, readIndex)) {\n      continue;\n    }\n\n    const current = arr[readIndex] as T;\n    if (await predicate(current, readIndex, arr)) {\n      // eslint-disable-next-line require-atomic-updates -- Yes, it is a potential race condition, but I don't an elegant way to fix it.\n      arr[writeIndex++] = current;\n    }\n  }\n  arr.length = writeIndex;\n}\n\n/**\n * Maps over an array asynchronously, applying the provided callback function to each element, and then flattens the results into a single array.\n *\n * @typeParam T - The type of elements in the input array.\n * @typeParam U - The type of elements in the output array.\n * @param arr - The array to map over and flatten.\n * @param callback - The callback function to apply to each element.\n * @returns A {@link Promise} that resolves with a flattened array of the results of the callback function.\n */\nexport async function asyncFlatMap<T, U>(arr: T[], callback: (value: T, index: number, array: T[]) => Promisable<U[]>): Promise<U[]> {\n  return (await asyncMap(arr, callback)).flat();\n}\n\n/**\n * Maps over an array asynchronously, applying the provided callback function to each element.\n *\n * @typeParam T - The type of elements in the input array.\n * @typeParam U - The type of elements in the output array.\n * @param arr - The array to map over.\n * @param callback - The callback function to apply to each element.\n * @returns A {@link Promise} that resolves with an array of the results of the callback function.\n */\nexport async function asyncMap<T, U>(arr: T[], callback: (value: T, index: number, array: T[]) => Promisable<U>): Promise<U[]> {\n  return await promiseAllSequentially(arr.map(callback));\n}\n\n/**\n * Converts an asynchronous function to a synchronous one by automatically handling the Promise rejection.\n *\n * @typeParam Args - The types of the arguments the function accepts.\n * @param asyncFunc - The asynchronous function to convert.\n * @param stackTrace - The stack trace of the source function.\n * @returns A function that wraps the asynchronous function in a synchronous interface.\n */\nexport function convertAsyncToSync<Args extends unknown[]>(asyncFunc: (...args: Args) => Promise<unknown>, stackTrace?: string): (...args: Args) => void {\n  stackTrace ??= getStackTrace(1);\n  return (...args: Args): void => {\n    const innerStackTrace = getStackTrace(1);\n    stackTrace = `${stackTrace ?? ''}\\n    at --- convertAsyncToSync --- (0)\\n${innerStackTrace}`;\n    invokeAsyncSafely(() => asyncFunc(...args), stackTrace);\n  };\n}\n\n/**\n * Converts a synchronous function to an asynchronous one by wrapping it in a {@link Promise}.\n *\n * @typeParam Args - The types of the arguments the function accepts.\n * @typeParam Result - The type of the function's return value.\n * @param syncFn - The synchronous function to convert.\n * @returns A function that wraps the synchronous function in an asynchronous interface.\n */\nexport function convertSyncToAsync<Args extends unknown[], Result>(syncFn: (...args: Args) => Result): (...args: Args) => Promise<Result> {\n  return async (...args: Args): Promise<Result> => {\n    await Promise.resolve();\n    return syncFn(...args);\n  };\n}\n\n/**\n * Handles a silent error.\n *\n * @param error - The error to handle.\n * @returns Whether the error is a silent error.\n */\nexport function handleSilentError(error: unknown): boolean {\n  let cause = error;\n  while (!(cause instanceof SilentError)) {\n    if (!(cause instanceof Error)) {\n      return false;\n    }\n\n    cause = cause.cause;\n  }\n\n  getLibDebugger('Async:handleSilentError')(error);\n  return true;\n}\n\n/**\n * Ignores an error that is thrown by an asynchronous function.\n *\n * @param promise - The promise to ignore the error of.\n * @param fallbackValue - Always `undefined`.\n * @returns A {@link Promise} that resolves when the asynchronous function completes or fails.\n */\nexport async function ignoreError(promise: Promise<unknown>, fallbackValue?: undefined): Promise<void>;\n\n/**\n * Invokes an asynchronous function and returns a fallback value if an error is thrown.\n *\n * @typeParam T - The type of the value returned by the asynchronous function.\n * @param promise - The promise to ignore the error of.\n * @param fallbackValue - The value to return if an error is thrown.\n * @returns A {@link Promise} that resolves with the value returned by the asynchronous function or the fallback value if an error is thrown.\n */\nexport async function ignoreError<T>(promise: Promise<T>, fallbackValue: T): Promise<T> {\n  const ignoreErrorDebugger = getLibDebugger('Async:ignoreError');\n  const stackTrace = getStackTrace(1);\n  try {\n    return await promise;\n  } catch (e) {\n    ignoreErrorDebugger('Ignored error', new CustomStackTraceError('Ignored error', stackTrace, e));\n    return fallbackValue;\n  }\n}\n\n/**\n * Invokes a {@link Promise} and safely handles any errors by catching them and emitting an async error event.\n *\n * @param asyncFn - The asynchronous function to invoke safely.\n * @param stackTrace - The stack trace of the source function.\n */\nexport function invokeAsyncSafely(asyncFn: () => Promise<unknown>, stackTrace?: string): void {\n  stackTrace ??= getStackTrace(1);\n  // eslint-disable-next-line no-void -- We need to fire-and-forget.\n  void addErrorHandler(asyncFn, stackTrace);\n}\n\n/**\n * Invokes an asynchronous function after a delay.\n *\n * @param asyncFn - The asynchronous function to invoke.\n * @param delayInMilliseconds - The delay in milliseconds.\n * @param stackTrace - The stack trace of the source function.\n * @param abortSignal - The abort signal to listen to.\n */\nexport function invokeAsyncSafelyAfterDelay(\n  asyncFn: (abortSignal: AbortSignal) => Promisable<void>,\n  delayInMilliseconds = 0,\n  stackTrace?: string,\n  abortSignal?: AbortSignal\n): void {\n  abortSignal ??= abortSignalNever();\n  abortSignal.throwIfAborted();\n  stackTrace ??= getStackTrace(1);\n  invokeAsyncSafely(async () => {\n    await sleep(delayInMilliseconds, abortSignal, true);\n    await asyncFn(abortSignal);\n  }, stackTrace);\n}\n\n/**\n * Executes async functions sequentially.\n *\n * @typeParam T - The type of the value.\n * @param asyncFns - The async functions to execute sequentially.\n * @returns A {@link Promise} that resolves with an array of the results of the async functions.\n */\nexport async function promiseAllAsyncFnsSequentially<T>(asyncFns: (() => Promisable<T>)[]): Promise<T[]> {\n  const results: T[] = [];\n  for (const asyncFn of asyncFns) {\n    results.push(await asyncFn());\n  }\n  return results;\n}\n\n/**\n * Executes promises sequentially.\n *\n * @typeParam T - The type of the value.\n * @param promises - The promises to execute sequentially.\n * @returns A {@link Promise} that resolves with an array of the results of the promises.\n */\nexport async function promiseAllSequentially<T>(promises: Promisable<T>[]): Promise<T[]> {\n  return await promiseAllAsyncFnsSequentially(promises.map((promise) => () => promise));\n}\n\nconst terminateRetryErrors = new WeakSet<Error>();\n\n/**\n * Options for {@link retryWithTimeout}.\n */\nexport interface RetryWithTimeoutOptions {\n  /**\n   * The function to handle the timeout.\n   *\n   * @param context - The timeout context.\n   */\n  onTimeout?(this: void, context: TimeoutContext): void;\n\n  /**\n   * The function to execute.\n   *\n   * @param abortSignal - The abort signal to listen to.\n   * @returns The result of the function.\n   */\n  operationFn(this: void, abortSignal: AbortSignal): Promisable<boolean>;\n\n  /**\n   * The name of the operation.\n   */\n  operationName?: string;\n\n  /**\n   * The retry options.\n   */\n  retryOptions?: RetryOptions;\n\n  /**\n   * The stack trace of the source function.\n   */\n  stackTrace?: string;\n}\n\n/**\n * Options for {@link runWithTimeout}.\n */\nexport interface RunWithTimeoutOptions<Result> {\n  /**\n   * The context of the function.\n   */\n  context?: unknown;\n\n  /**\n   * The function to handle the timeout.\n   *\n   * @param context - The timeout context.\n   */\n  onTimeout?(this: void, context: TimeoutContext): void;\n\n  /**\n   * The operation function to execute.\n   *\n   * @param abortSignal - The abort signal to listen to.\n   * @returns The result of the function.\n   */\n  operationFn(this: void, abortSignal: AbortSignal): Promisable<Result>;\n\n  /**\n   * The name of the operation.\n   */\n  operationName?: string;\n\n  /**\n   * The stack trace of the source function.\n   */\n  stackTrace?: string | undefined;\n\n  /**\n   * The maximum time to wait in milliseconds.\n   */\n  timeoutInMilliseconds: number;\n}\n\n/**\n * Context provided to the timeout handler.\n */\nexport interface TimeoutContext {\n  /**\n   * The duration in milliseconds since the operation started.\n   */\n  duration: number;\n  /**\n   * Registers a callback to be invoked when the operation completes.\n   *\n   * @param callback - The function to call when the operation completes.\n   */\n  onOperationCompleted(callback: () => void): void;\n  /**\n   * The name of the operation.\n   */\n  operationName: string;\n  /**\n   * Terminates the operation that timed out.\n   */\n  terminateOperation(): void;\n}\n\n/**\n * Marks an error to terminate retry logic.\n *\n * @param error - The error to mark to terminate retry logic.\n */\nexport function marksAsTerminateRetry(error: Error): void {\n  terminateRetryErrors.add(error);\n}\n\n/**\n * An async function that never ends.\n *\n * @returns A {@link Promise} that never resolves.\n */\nexport async function neverEnds(): Promise<never> {\n  await new Promise(() => {\n    noop();\n  });\n  throw new Error('Should never happen');\n}\n\n/**\n * Gets the next tick.\n *\n * @returns A promise that resolves when the next tick is available.\n */\nexport async function nextTickAsync(): Promise<void> {\n  return new Promise((resolve) => {\n    process.nextTick(() => {\n      resolve();\n    });\n  });\n}\n\n/**\n * Gets the next queue microtask.\n *\n * @returns A promise that resolves when the next queue microtask is available.\n */\nexport async function queueMicrotaskAsync(): Promise<void> {\n  return new Promise((resolve) => {\n    queueMicrotask(() => {\n      resolve();\n    });\n  });\n}\n\n/**\n * Gets the next request animation frame.\n *\n * @returns A promise that resolves when the next request animation frame is available.\n */\nexport async function requestAnimationFrameAsync(): Promise<void> {\n  return new Promise((resolve) => {\n    requestAnimationFrame(() => {\n      resolve();\n    });\n  });\n}\n\n/**\n * Retries the provided function until it returns true or the timeout is reached.\n *\n * @param options - The options for the function.\n * @returns A {@link Promise} that resolves when the function returns true or rejects when the timeout is reached.\n */\nexport async function retryWithTimeout(options: RetryWithTimeoutOptions): Promise<void> {\n  const retryWithTimeoutDebugger = getLibDebugger('Async:retryWithTimeout');\n  const stackTrace = options.stackTrace ?? getStackTrace(1);\n  const DEFAULT_RETRY_OPTIONS = {\n    // eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.\n    retryDelayInMilliseconds: 100,\n    shouldRetryOnError: false,\n    // eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.\n    timeoutInMilliseconds: 5000\n  };\n  const fullOptions = { ...DEFAULT_RETRY_OPTIONS, ...options.retryOptions };\n  fullOptions.abortSignal?.throwIfAborted();\n\n  await runWithTimeout(normalizeOptionalProperties<RunWithTimeoutOptions<void>>({\n    context: { operationName: options.operationName ?? '', retryFn: options.operationFn },\n    onTimeout: options.onTimeout,\n    async operationFn(abortSignal: AbortSignal): Promise<void> {\n      const combinedAbortSignal = abortSignalAny(fullOptions.abortSignal, abortSignal);\n      combinedAbortSignal.throwIfAborted();\n      let attempt = 0;\n      while (!combinedAbortSignal.aborted) {\n        attempt++;\n        let isSuccess: boolean;\n        try {\n          isSuccess = await options.operationFn(combinedAbortSignal);\n        } catch (error) {\n          // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition -- It might changed inside `fn()`. ESLint mistakenly does not recognize it.\n          if (combinedAbortSignal.aborted || !fullOptions.shouldRetryOnError || terminateRetryErrors.has(error as Error)) {\n            throw new CustomStackTraceError('retryWithTimeout failed', stackTrace, error);\n          }\n          printError(error);\n          isSuccess = false;\n        }\n        if (isSuccess) {\n          printWithStackTrace(retryWithTimeoutDebugger, stackTrace, `Retry completed successfully after ${String(attempt)} attempts`, {\n            operationFn: options.operationFn,\n            operationName: options.operationName ?? ''\n          });\n          return;\n        }\n\n        printWithStackTrace(\n          retryWithTimeoutDebugger,\n          stackTrace,\n          `Retry attempt ${String(attempt)} completed unsuccessfully. Trying again in ${String(fullOptions.retryDelayInMilliseconds)} milliseconds`,\n          {\n            operationFn: options.operationFn,\n            operationName: options.operationName ?? ''\n          }\n        );\n\n        await sleep(fullOptions.retryDelayInMilliseconds, abortSignal);\n      }\n    },\n    operationName: options.operationName ?? '',\n    stackTrace,\n    timeoutInMilliseconds: fullOptions.timeoutInMilliseconds\n  }));\n}\n\n/**\n * Executes a function with a timeout. If the function does not complete within the specified time, it is considered to have timed out.\n *\n * If `DEBUG=obsidian-dev-utils:Async:runWithTimeout` is set, the execution is not terminated after the timeout and the function is allowed to run indefinitely.\n *\n * @typeParam Result - The type of the result from the asynchronous function.\n * @param options - The options for the function.\n * @returns A {@link Promise} that resolves with the result of the asynchronous function or rejects if it times out.\n */\nexport async function runWithTimeout<Result>(options: RunWithTimeoutOptions<Result>): Promise<Result> {\n  const stackTrace = options.stackTrace ?? getStackTrace(1);\n  const startTime = performance.now();\n\n  const runAbortController = new AbortController();\n  const timeoutAbortController = new AbortController();\n\n  let result: null | Result = null;\n  let hasResult = false;\n  let isCompleted = false;\n  const runWithTimeoutDebugger = getLibDebugger('Async:runWithTimeout');\n  const onTimeout = options.onTimeout ?? defaultOnTimeout;\n\n  await Promise.race([run(), innerTimeout()]);\n  // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition -- It might changed inside `run()`. ESLint mistakenly does not recognize it.\n  if (hasResult) {\n    return result as Result;\n  }\n\n  throw new CustomStackTraceError('Run with timeout failed', stackTrace, runAbortController.signal.reason);\n\n  async function run(): Promise<void> {\n    try {\n      result = await options.operationFn(runAbortController.signal);\n      const duration = Math.trunc(performance.now() - startTime);\n      printWithStackTrace(runWithTimeoutDebugger, stackTrace, `Execution time: ${String(duration)} milliseconds`, {\n        context: options.context,\n        operationFn: options.operationFn,\n        operationName: options.operationName ?? ''\n      });\n      hasResult = true;\n    } catch (e) {\n      runAbortController.abort(e);\n    } finally {\n      isCompleted = true;\n      timeoutAbortController.abort(new Error('Completed'));\n    }\n  }\n\n  async function innerTimeout(): Promise<void> {\n    await sleep(options.timeoutInMilliseconds, timeoutAbortController.signal);\n\n    if (isCompleted) {\n      return;\n    }\n    const duration = Math.trunc(performance.now() - startTime);\n    printWithStackTrace(runWithTimeoutDebugger, stackTrace, `Timed out after ${String(duration)} milliseconds`, {\n      context: options.context,\n      operationFn: options.operationFn,\n      operationName: options.operationName ?? ''\n    });\n\n    const timeoutContext: TimeoutContext = normalizeOptionalProperties<TimeoutContext>({\n      duration,\n      onOperationCompleted(callback) {\n        timeoutAbortController.signal.addEventListener('abort', callback);\n      },\n      operationName: options.operationName ?? '',\n      terminateOperation() {\n        const error = new Error(`Timed out after ${String(duration)} milliseconds`);\n        runAbortController.abort(error);\n        timeoutAbortController.abort(error);\n      }\n    });\n\n    onTimeout(timeoutContext);\n    await waitForAbort(timeoutAbortController.signal);\n  }\n\n  function defaultOnTimeout(ctx: TimeoutContext): void {\n    ctx.terminateOperation();\n  }\n}\n\n/**\n * Gets the next set immediate.\n *\n * @returns A promise that resolves when the next set immediate is available.\n */\nexport async function setImmediateAsync(): Promise<void> {\n  return new Promise((resolve) => {\n    setImmediate(() => {\n      resolve();\n    });\n  });\n}\n\n/**\n * Delays execution for a specified number of milliseconds.\n *\n * @param delay - The time to wait in milliseconds.\n * @returns A {@link Promise} that resolves after the specified delay.\n */\nexport async function setTimeoutAsync(delay?: number): Promise<void> {\n  await new Promise((resolve) => {\n    setTimeout(resolve, delay);\n  });\n}\n\n/**\n * Delays execution for a specified number of milliseconds.\n *\n * @param milliseconds - The time to wait in milliseconds.\n * @param abortSignal - The abort signal to listen to.\n * @param shouldThrowOnAbort - Whether to throw an error if the abort signal is aborted.\n * @returns A {@link Promise} that resolves after the specified delay.\n */\nexport async function sleep(milliseconds: number, abortSignal?: AbortSignal, shouldThrowOnAbort?: boolean): Promise<void> {\n  await waitForAbort(abortSignalAny(abortSignal, abortSignalTimeout(milliseconds)));\n  if (shouldThrowOnAbort) {\n    abortSignal?.throwIfAborted();\n  }\n}\n\n/**\n * Returns a {@link Promise} that rejects after the specified timeout period.\n *\n * @param timeoutInMilliseconds - The timeout period in milliseconds.\n * @param abortSignal - The abort signal to listen to.\n * @param shouldThrowOnAbort - Whether to throw an error if the abort signal is aborted.\n * @returns A {@link Promise} that always rejects with a timeout error.\n */\nexport async function timeout(timeoutInMilliseconds: number, abortSignal?: AbortSignal, shouldThrowOnAbort?: boolean): Promise<never> {\n  await sleep(timeoutInMilliseconds, abortSignal, shouldThrowOnAbort);\n  throw new Error(`Timed out in ${String(timeoutInMilliseconds)} milliseconds`);\n}\n\n/**\n * Converts an AsyncIterableIterator to an array by consuming all its elements.\n *\n * @typeParam T - The type of elements produced by the AsyncIterableIterator.\n * @param iter - The AsyncIterableIterator to convert.\n * @returns A {@link Promise} that resolves with an array of all the elements in the AsyncIterableIterator.\n */\nexport async function toArray<T>(iter: AsyncIterableIterator<T>): Promise<T[]> {\n  const arr: T[] = [];\n  for await (const item of iter) {\n    arr.push(item);\n  }\n  return arr;\n}\n"],
  "mappings": "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