obsidian-dev-utils 38.10.0 → 39.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +10 -0
- package/dist/lib/cjs/Async.cjs +16 -22
- package/dist/lib/cjs/Library.cjs +1 -1
- package/dist/lib/cjs/ScriptUtils/ESLint/eslint.config.cjs +3 -2
- package/dist/lib/cjs/obsidian/FileChange.cjs +103 -87
- package/dist/lib/cjs/obsidian/FileChange.d.cts +2 -2
- package/dist/lib/cjs/obsidian/Link.cjs +10 -3
- package/dist/lib/esm/Async.mjs +16 -22
- package/dist/lib/esm/Library.mjs +1 -1
- package/dist/lib/esm/ScriptUtils/ESLint/eslint.config.mjs +3 -2
- package/dist/lib/esm/obsidian/FileChange.d.mts +2 -2
- package/dist/lib/esm/obsidian/FileChange.mjs +103 -87
- package/dist/lib/esm/obsidian/Link.mjs +10 -3
- package/package.json +1 -1
|
@@ -115,9 +115,16 @@ async function editBacklinks(app, pathOrFile, linkConverter, processOptions = {}
|
|
|
115
115
|
}
|
|
116
116
|
}
|
|
117
117
|
async function editLinks(app, pathOrFile, linkConverter, processOptions = {}) {
|
|
118
|
-
await (0, import_FileChange.applyFileChanges)(app, pathOrFile, async () => {
|
|
118
|
+
await (0, import_FileChange.applyFileChanges)(app, pathOrFile, async (abortSignal, content) => {
|
|
119
119
|
const cache = await (0, import_MetadataCache.getCacheSafe)(app, pathOrFile);
|
|
120
|
-
|
|
120
|
+
abortSignal.throwIfAborted();
|
|
121
|
+
const file = (0, import_FileSystem.getFile)(app, pathOrFile);
|
|
122
|
+
const cachedContent = await app.vault.cachedRead(file);
|
|
123
|
+
abortSignal.throwIfAborted();
|
|
124
|
+
if (content !== cachedContent) {
|
|
125
|
+
return null;
|
|
126
|
+
}
|
|
127
|
+
return await getFileChanges(cache, (0, import_FileSystem.isCanvasFile)(app, pathOrFile), linkConverter, abortSignal);
|
|
121
128
|
}, processOptions);
|
|
122
129
|
}
|
|
123
130
|
async function editLinksInContent(app, content, linkConverter, abortSignal) {
|
|
@@ -637,4 +644,4 @@ function shouldUseWikilinkStyle(app, originalLink, linkStyle) {
|
|
|
637
644
|
updateLinksInContent,
|
|
638
645
|
updateLinksInFile
|
|
639
646
|
});
|
|
640
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Link.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utilities for handling and updating links within Obsidian vaults. It includes\n * functions to split paths, update links in files, and generate markdown links with various options.\n */\n\nimport type {\n  Link,\n  Text\n} from 'mdast';\nimport type {\n  App,\n  CachedMetadata,\n  Reference,\n  TFile\n} from 'obsidian';\nimport type { Promisable } from 'type-fest';\nimport type { Node } from 'unist';\n\nimport {\n  normalizePath,\n  parseLinktext\n} from 'obsidian';\nimport { InternalPluginName } from 'obsidian-typings/implementations';\nimport { remark } from 'remark';\nimport remarkParse from 'remark-parse';\nimport { wikiLinkPlugin } from 'remark-wiki-link';\nimport { visit } from 'unist-util-visit';\n\nimport type { GenericObject } from '../ObjectUtils.ts';\nimport type { MaybeReturn } from '../Type.ts';\nimport type { FileChange } from './FileChange.ts';\nimport type { PathOrFile } from './FileSystem.ts';\nimport type { ProcessOptions } from './Vault.ts';\n\nimport { abortSignalNever } from '../AbortController.ts';\nimport {\n  normalizeOptionalProperties,\n  toJson\n} from '../ObjectUtils.ts';\nimport {\n  basename,\n  dirname,\n  extname,\n  join,\n  relative\n} from '../Path.ts';\nimport {\n  normalize,\n  replaceAll\n} from '../String.ts';\nimport { isUrl } from '../url.ts';\nimport {\n  applyContentChanges,\n  applyFileChanges,\n  isCanvasChange,\n  isContentChange\n} from './FileChange.ts';\nimport {\n  getFile,\n  getPath,\n  isCanvasFile,\n  isMarkdownFile,\n  trimMarkdownExtension\n} from './FileSystem.ts';\nimport {\n  getAllLinks,\n  getBacklinksForFileSafe,\n  getCacheSafe,\n  parseMetadata,\n  tempRegisterFilesAndRun\n} from './MetadataCache.ts';\nimport {\n  shouldUseRelativeLinks,\n  shouldUseWikilinks\n} from './ObsidianSettings.ts';\nimport {\n  isCanvasFileNodeReference,\n  referenceToFileChange\n} from './Reference.ts';\n\nconst ESCAPED_WIKILINK_DIVIDER = '\\\\|';\n\n/**\n * Regular expression for special link symbols.\n */\n// eslint-disable-next-line no-control-regex\nconst SPECIAL_LINK_SYMBOLS_REGEXP = /[\\\\\\x00\\x08\\x0B\\x0C\\x0E-\\x1F ]/g;\n\n/**\n * Regular expression for special markdown link symbols.\n */\nconst SPECIAL_MARKDOWN_LINK_SYMBOLS_REGEX = /[\\\\[\\]<>_*~=`$]/g;\n\n/**\n * Regular expression for unescaped pipes.\n */\nconst UNESCAPED_WIKILINK_DIVIDER_REGEXP = /(?<!\\\\)\\|/g;\n\nconst WIKILINK_DIVIDER = '|';\n\n/**\n * A style of the link.\n */\nexport enum LinkStyle {\n  /**\n   * Force the link to be in markdown format.\n   */\n  Markdown = 'Markdown',\n\n  /**\n   * Use the default link style defined in Obsidian settings.\n   */\n  ObsidianSettingsDefault = 'ObsidianSettingsDefault',\n\n  /**\n   * Preserve the existing link style.\n   */\n  PreserveExisting = 'PreserveExisting',\n\n  /**\n   * Force the link to be in wikilink format.\n   */\n  Wikilink = 'Wikilink'\n}\n\n/**\n * Options for {@link convertLink}.\n */\nexport interface ConvertLinkOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A reference for the link.\n   */\n  link: Reference;\n\n  /**\n   * A style of the link.\n   */\n  linkStyle?: LinkStyle;\n\n  /**\n   * A source file containing the link.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * An old path of the link.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * Whether to update file name alias. Defaults to `true`.\n   */\n  shouldUpdateFileNameAlias?: boolean;\n}\n\n/**\n * Wrapper for default options for {@link GenerateMarkdownLinkOptions}.\n */\nexport interface GenerateMarkdownLinkDefaultOptionsWrapper {\n  /**\n   * A default options for generating markdown links.\n   *\n   * @returns A default options for generating markdown links.\n   */\n  defaultOptionsFn(): Partial<GenerateMarkdownLinkOptions>;\n}\n\n/**\n * Options for {@link generateMarkdownLink}.\n */\nexport interface GenerateMarkdownLinkOptions {\n  /**\n   * An alias for the link.\n   */\n  alias?: string;\n\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * Indicates if the link should be embedded. If not provided, it will be inferred based on the file type.\n   */\n  isEmbed?: boolean;\n\n  /**\n   * Whether to allow an empty alias for embeds. Defaults to `true`.\n   */\n  isEmptyEmbedAliasAllowed?: boolean;\n\n  /**\n   * Whether to allow non-existing files. If `false` and `pathOrFile` is a non-existing file, an error will be thrown. Defaults to `false`.\n   */\n  isNonExistingFileAllowed?: boolean;\n\n  /**\n   * A style of the link.\n   */\n  linkStyle?: LinkStyle;\n\n  /**\n   * An original link text. If provided, it will be used to infer the values of `isEmbed`, `isWikilink`, `useLeadingDot`, and `useAngleBrackets`.\n   * These inferred values will be overridden by corresponding settings if specified.\n   */\n  originalLink?: string;\n\n  /**\n   * Indicates if the link should be relative. If not provided or `false`, it will be inferred based on the Obsidian settings.\n   */\n  shouldForceRelativePath?: boolean;\n\n  /**\n   * Whether to include the attachment extension in the embed alias. Has no effect if `allowEmptyEmbedAlias` is `true`. Defaults to `false`.\n   */\n  shouldIncludeAttachmentExtensionToEmbedAlias?: boolean;\n\n  /**\n   * Indicates if the link should use angle brackets. Defaults to `false`. Has no effect if `isWikilink` is `true`\n   */\n  shouldUseAngleBrackets?: boolean;\n\n  /**\n   * Indicates if the link should use a leading dot. Defaults to `false`. Has no effect if `isWikilink` is `true` or `isRelative` is `false`.\n   */\n  shouldUseLeadingDot?: boolean;\n\n  /**\n   * A source path of the link.\n   */\n  sourcePathOrFile: PathOrFile;\n\n  /**\n   * A subpath of the link.\n   */\n  subpath?: string;\n\n  /**\n   * A target path or file.\n   */\n  targetPathOrFile: PathOrFile;\n}\n\n/**\n * A result of parsing a link.\n */\nexport interface ParseLinkResult {\n  /**\n   * An alias of the link.\n   */\n  alias?: string;\n\n  /**\n   * An encoded URL of the link.\n   */\n  encodedUrl?: string;\n\n  /**\n   * An end offset of the link in the original text.\n   */\n  endOffset: number;\n\n  /**\n   * Indicates if the link has angle brackets.\n   */\n  hasAngleBrackets?: boolean;\n\n  /**\n   * Indicates if the link is an embed link.\n   */\n  isEmbed: boolean;\n\n  /**\n   * Indicates if the link is external.\n   */\n  isExternal: boolean;\n\n  /**\n   * Indicates if the link is a wikilink.\n   */\n  isWikilink: boolean;\n\n  /**\n   * A raw link text.\n   */\n  raw: string;\n\n  /**\n   * A start offset of the link in the original text.\n   */\n  startOffset: number;\n\n  /**\n   * A title of the link.\n   */\n  title?: string;\n\n  /**\n   * An URL of the link.\n   */\n  url: string;\n}\n\n/**\n * Options for {@link shouldResetAlias}.\n */\nexport interface ShouldResetAliasOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A display text of the link.\n   */\n  displayText: string | undefined;\n\n  /**\n   * Indicates if the link is a wikilink.\n   */\n  isWikilink?: boolean;\n\n  /**\n   * A source path of the link.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * An old source file containing the link.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * An old target path of the link.\n   */\n  oldTargetPath: PathOrFile;\n\n  /**\n   * A target path or file.\n   */\n  targetPathOrFile: PathOrFile;\n}\n\n/**\n * Splits a link into its link path and subpath.\n */\nexport interface SplitSubpathResult {\n  /**\n   * A link path.\n   */\n  linkPath: string;\n\n  /**\n   * A subpath.\n   */\n  subpath: string;\n}\n\n/**\n * Options for {@link updateLink}.\n */\nexport interface UpdateLinkOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A reference for the link.\n   */\n  link: Reference;\n\n  /**\n   * Whether to force markdown links.\n   */\n  linkStyle?: LinkStyle;\n\n  /**\n   * A source file containing the link.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * A file associated with the link.\n   */\n  newTargetPathOrFile: PathOrFile;\n\n  /**\n   * An old source file containing the link.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * An old path of the file.\n   */\n  oldTargetPathOrFile?: PathOrFile;\n\n  /**\n   * Whether to update file name alias. Defaults to `true`.\n   */\n  shouldUpdateFileNameAlias?: boolean;\n}\n\n/**\n * Options for {@link updateLinksInFile}.\n */\nexport interface UpdateLinksInFileOptions extends ProcessOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A style of the link.\n   */\n  linkStyle?: LinkStyle;\n\n  /**\n   * A file to update the links in.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * An old path of the file.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * Whether to update only embedded links.\n   */\n  shouldUpdateEmbedOnlyLinks?: boolean;\n\n  /**\n   * Whether to update file name alias. Defaults to `true`.\n   */\n  shouldUpdateFileNameAlias?: boolean;\n}\n\ninterface LinkConfig {\n  isEmbed: boolean;\n  isWikilink: boolean;\n  shouldForceRelativePath: boolean;\n  shouldUseAngleBrackets: boolean;\n  shouldUseLeadingDot: boolean;\n}\n\ninterface TablePosition {\n  end: number;\n  start: number;\n}\n\n/**\n * Options for {@link updateLinksInContent}.\n */\ninterface UpdateLinksInContentOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A content to update the links in.\n   */\n  content: string;\n\n  /**\n   * A style of the link.\n   */\n  linkStyle?: LinkStyle;\n\n  /**\n   * A new source path or file.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * An old source path or file.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * Whether to update only embedded links.\n   */\n  shouldUpdateEmbedOnlyLinks?: boolean;\n\n  /**\n   * Whether to update file name alias.\n   */\n  shouldUpdateFileNameAlias?: boolean;\n}\n\ninterface WikiLinkNode extends Node {\n  data: {\n    alias: string;\n  };\n  value: string;\n}\n\n/**\n * Converts a link to a new path.\n *\n * @param options - The options for converting the link.\n * @returns The converted link.\n */\nexport function convertLink(options: ConvertLinkOptions): string {\n  const targetFile = extractLinkFile(options.app, options.link, options.oldSourcePathOrFile ?? options.newSourcePathOrFile);\n  if (!targetFile) {\n    return options.link.original;\n  }\n\n  return updateLink(normalizeOptionalProperties<UpdateLinkOptions>({\n    app: options.app,\n    link: options.link,\n    linkStyle: options.linkStyle,\n    newSourcePathOrFile: options.newSourcePathOrFile,\n    newTargetPathOrFile: targetFile,\n    oldSourcePathOrFile: options.oldSourcePathOrFile,\n    shouldUpdateFileNameAlias: options.shouldUpdateFileNameAlias\n  }));\n}\n\n/**\n * Edits the backlinks for a file or path.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or file to edit the backlinks for.\n * @param linkConverter - The function that converts each link.\n * @param processOptions - Optional options for retrying the operation.\n * @returns A {@link Promise} that resolves when the backlinks have been edited.\n */\nexport async function editBacklinks(\n  app: App,\n  pathOrFile: PathOrFile,\n  linkConverter: (link: Reference) => Promisable<MaybeReturn<string>>,\n  processOptions: ProcessOptions = {}\n): Promise<void> {\n  const backlinks = await getBacklinksForFileSafe(app, pathOrFile, processOptions);\n  for (const backlinkNotePath of backlinks.keys()) {\n    const currentLinks = backlinks.get(backlinkNotePath) ?? [];\n    const linkJsons = new Set<string>(currentLinks.map((link) => toJson(link)));\n    await editLinks(app, backlinkNotePath, (link) => {\n      const linkJson = toJson(link);\n      if (!linkJsons.has(linkJson)) {\n        return;\n      }\n\n      return linkConverter(link);\n    }, processOptions);\n  }\n}\n\n/**\n * Edits the backlinks for a file or path.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or file to edit the backlinks for.\n * @param linkConverter - The function that converts each link.\n * @param processOptions - Optional options for retrying the operation.\n * @returns A {@link Promise} that resolves when the backlinks have been edited.\n */\nexport async function editLinks(\n  app: App,\n  pathOrFile: PathOrFile,\n  linkConverter: (link: Reference) => Promisable<MaybeReturn<string>>,\n  processOptions: ProcessOptions = {}\n): Promise<void> {\n  await applyFileChanges(app, pathOrFile, async () => {\n    const cache = await getCacheSafe(app, pathOrFile);\n    return await getFileChanges(cache, isCanvasFile(app, pathOrFile), linkConverter);\n  }, processOptions);\n}\n\n/**\n * Edits the links in a content string.\n *\n * @param app - The Obsidian application instance.\n * @param content - The content to edit the links in.\n * @param linkConverter - The function that converts each link.\n * @param abortSignal - The abort signal to control the execution of the function.\n * @returns The promise that resolves to the updated content.\n */\nexport async function editLinksInContent(\n  app: App,\n  content: string,\n  linkConverter: (link: Reference) => Promisable<MaybeReturn<string>>,\n  abortSignal?: AbortSignal\n): Promise<string> {\n  abortSignal ??= abortSignalNever();\n  abortSignal.throwIfAborted();\n\n  const newContent = await applyContentChanges(abortSignal, content, '', async () => {\n    const cache = await parseMetadata(app, content);\n    abortSignal.throwIfAborted();\n    const changes = await getFileChanges(cache, false, linkConverter, abortSignal);\n    abortSignal.throwIfAborted();\n    return changes;\n  });\n  abortSignal.throwIfAborted();\n\n  if (newContent === null) {\n    throw new Error('Failed to update links in content');\n  }\n\n  return newContent;\n}\n\n/**\n * Encodes a URL.\n *\n * @param url - The URL to encode.\n * @returns The encoded URL.\n */\nexport function encodeUrl(url: string): string {\n  return replaceAll(url, SPECIAL_LINK_SYMBOLS_REGEXP, ({ substring: specialLinkSymbol }) => encodeURIComponent(specialLinkSymbol));\n}\n\n/**\n * Extracts the file associated with a link.\n *\n * @param app - The Obsidian application instance.\n * @param link - The reference cache for the link.\n * @param sourcePathOrFile - The source path or file.\n * @param shouldAllowNonExistingFile - Whether to allow non-existing files. Defaults to `false`.\n * @returns The file associated with the link, or null if not found.\n */\nexport function extractLinkFile(app: App, link: Reference, sourcePathOrFile: PathOrFile, shouldAllowNonExistingFile = false): null | TFile {\n  const { linkPath } = splitSubpath(link.link);\n  const sourcePath = getPath(app, sourcePathOrFile);\n  const file = app.metadataCache.getFirstLinkpathDest(linkPath, sourcePath);\n  if (file) {\n    return file;\n  }\n\n  if (!shouldAllowNonExistingFile) {\n    return null;\n  }\n\n  if (linkPath.startsWith('/')) {\n    return getFile(app, linkPath, true);\n  }\n\n  const fullLinkPath = join(dirname(sourcePath), `./${linkPath}`);\n\n  if (fullLinkPath.startsWith('../')) {\n    return null;\n  }\n\n  return getFile(app, fullLinkPath, true);\n}\n\n/**\n * Fixes the frontmatter markdown links in the provided metadata cache.\n *\n * @param cache - The metadata cache to fix the frontmatter markdown links in.\n * @returns Whether the frontmatter markdown links were fixed.\n */\nexport function fixFrontmatterMarkdownLinks(cache: CachedMetadata): boolean {\n  return _fixFrontmatterMarkdownLinks(cache.frontmatter, '', cache);\n}\n\n/**\n * Generates a markdown link based on the provided parameters.\n *\n * @param options - The options for generating the markdown link.\n * @returns The generated markdown link.\n */\nexport function generateMarkdownLink(options: GenerateMarkdownLinkOptions): string {\n  const { app } = options;\n\n  const configurableDefaultOptionsFn = (app.fileManager.generateMarkdownLink as Partial<GenerateMarkdownLinkDefaultOptionsWrapper>).defaultOptionsFn\n    ?? ((): Partial<GenerateMarkdownLinkOptions> => ({}));\n  const configurableDefaultOptions = configurableDefaultOptionsFn();\n\n  const DEFAULT_OPTIONS: Partial<GenerateMarkdownLinkOptions> = {\n    isEmptyEmbedAliasAllowed: true\n  };\n\n  options = { ...DEFAULT_OPTIONS, ...configurableDefaultOptions, ...options };\n\n  const targetFile = getFile(app, options.targetPathOrFile, options.isNonExistingFileAllowed);\n\n  return tempRegisterFilesAndRun(app, [targetFile], () => generateMarkdownLinkImpl(options));\n}\n\n/**\n * Parses a link into its components.\n *\n * @param str - The link to parse.\n * @returns The parsed link.\n */\nexport function parseLink(str: string): null | ParseLinkResult {\n  const links = parseLinks(str);\n  return links[0]?.raw === str ? links[0] : null;\n}\n\n/**\n * Parses all links in a string.\n *\n * @param str - The string to parse the links in.\n * @returns The parsed links.\n */\nexport function parseLinks(str: string): ParseLinkResult[] {\n  const embedSymbolOffsets = new Set<number>();\n\n  const EMBED_LINK_PREFIX = '![';\n  const NO_EMBED_LINK_PREFIX = ' [';\n\n  const noEmbedStr = replaceAll(str, EMBED_LINK_PREFIX, (args) => {\n    embedSymbolOffsets.add(args.offset);\n    return NO_EMBED_LINK_PREFIX;\n  });\n\n  const processor = remark().use(remarkParse).use(wikiLinkPlugin, { aliasDivider: WIKILINK_DIVIDER });\n  const root = processor.parse(noEmbedStr);\n\n  const links: ParseLinkResult[] = [];\n  const textLinks: ParseLinkResult[] = [];\n\n  visit(root, (node: Node) => {\n    let link: ParseLinkResult;\n    switch (node.type) {\n      case 'link':\n        link = parseLinkNode(node as Link, str);\n        break;\n      case 'wikiLink':\n        link = parseWikilinkNode(node as WikiLinkNode, str);\n        break;\n      default:\n        return;\n    }\n\n    if (embedSymbolOffsets.has(link.startOffset - 1)) {\n      link.isEmbed = true;\n      link.startOffset--;\n      link.raw = `!${link.raw}`;\n    }\n    links.push(link);\n  });\n\n  links.sort((a, b) => a.startOffset - b.startOffset);\n\n  let textStartOffset = 0;\n\n  for (const link of links) {\n    extractTextLinks(str, textStartOffset, link.startOffset - 1, textLinks);\n    textStartOffset = link.endOffset + 1;\n  }\n\n  extractTextLinks(str, textStartOffset, str.length - 1, textLinks);\n\n  links.push(...textLinks);\n  links.sort((a, b) => a.startOffset - b.startOffset);\n\n  return links;\n}\n\n/**\n * Determines if the alias of a link should be reset.\n *\n * @param options - The options for determining if the alias should be reset.\n * @returns Whether the alias should be reset.\n */\nexport function shouldResetAlias(options: ShouldResetAliasOptions): boolean {\n  const {\n    app,\n    displayText,\n    isWikilink,\n    newSourcePathOrFile,\n    oldSourcePathOrFile,\n    oldTargetPath,\n    targetPathOrFile\n  } = options;\n  if (isWikilink === false) {\n    return false;\n  }\n\n  if (!displayText) {\n    return true;\n  }\n\n  const targetFile = getFile(app, targetPathOrFile, true);\n  const newSourcePath = getPath(app, newSourcePathOrFile);\n  const oldSourcePath = getPath(app, oldSourcePathOrFile ?? newSourcePathOrFile);\n  const newSourceFolder = dirname(newSourcePath);\n  const oldSourceFolder = dirname(oldSourcePath);\n  const aliasesToReset = new Set<string>();\n\n  for (const pathOrFile of [targetFile.path, oldTargetPath]) {\n    if (!pathOrFile) {\n      continue;\n    }\n\n    const path = getPath(app, pathOrFile);\n    aliasesToReset.add(path);\n    aliasesToReset.add(basename(path));\n    aliasesToReset.add(relative(newSourceFolder, path));\n    aliasesToReset.add(relative(oldSourceFolder, path));\n  }\n\n  for (const sourcePath of [oldSourcePath, newSourcePath]) {\n    aliasesToReset.add(app.metadataCache.fileToLinktext(targetFile, sourcePath, false));\n  }\n\n  const cleanDisplayText = replaceAll(normalizePath(displayText.split(' > ')[0] ?? ''), /^\\.\\//g, '').toLowerCase();\n\n  for (const alias of aliasesToReset) {\n    if (alias.toLowerCase() === cleanDisplayText) {\n      return true;\n    }\n\n    const folder = dirname(alias);\n    const base = basename(alias, extname(alias));\n    if (join(folder, base).toLowerCase() === cleanDisplayText) {\n      return true;\n    }\n  }\n\n  return false;\n}\n\n/**\n * Splits a link into its link path and subpath.\n *\n * @param link - The link to split.\n * @returns An object containing the link path and subpath.\n */\nexport function splitSubpath(link: string): SplitSubpathResult {\n  const parsed = parseLinktext(normalize(link));\n  return {\n    linkPath: parsed.path,\n    subpath: parsed.subpath\n  };\n}\n\n/**\n * Tests whether a link uses angle brackets, possibly embed:\n * `[title](<link>)`, `![title](<link>)`.\n *\n * @param link - Link to test\n * @returns Whether the link uses angle brackets\n */\nexport function testAngleBrackets(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.hasAngleBrackets ?? false;\n}\n\n/**\n * Tests whether a link is an embed link:\n * `![[link]]`, `![title](link)`.\n *\n * @param link - Link to test\n * @returns Whether the link is an embed link\n */\nexport function testEmbed(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.isEmbed ?? false;\n}\n\n/**\n * Tests whether a link has a leading dot, possibly embed:\n * `[[./link]]`, `[title](./link)`, `[title](<./link>)`,\n * `![[./link]]`, `![title](./link)`, `![title](<./link>)`.\n *\n * @param link - Link to test\n * @returns Whether the link has a leading dot\n */\nexport function testLeadingDot(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.url.startsWith('./') ?? false;\n}\n\n/**\n * Tests whether a link is a wikilink, possibly embed:\n * `[[link]]`, `![[link]]`.\n *\n * @param link - Link to test\n * @returns Whether the link is a wikilink\n */\nexport function testWikilink(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.isWikilink ?? false;\n}\n\n/**\n * Updates a link based on the provided parameters.\n *\n * @param options - The options for updating the link.\n * @returns The updated link.\n */\nexport function updateLink(options: UpdateLinkOptions): string {\n  const {\n    app,\n    link,\n    linkStyle,\n    newSourcePathOrFile,\n    newTargetPathOrFile,\n    oldSourcePathOrFile,\n    oldTargetPathOrFile,\n    shouldUpdateFileNameAlias\n  } = options;\n  if (!newTargetPathOrFile) {\n    return link.original;\n  }\n  const targetFile = getFile(app, newTargetPathOrFile, true);\n  const oldTargetPath = getPath(app, oldTargetPathOrFile ?? newTargetPathOrFile);\n  const isWikilink = shouldUseWikilinkStyle(app, link.original, linkStyle);\n\n  const { subpath } = splitSubpath(link.link);\n  let shouldKeepAlias = !shouldUpdateFileNameAlias;\n\n  if (isCanvasFile(app, newSourcePathOrFile)) {\n    if (isCanvasFileNodeReference(link)) {\n      return targetFile.path + subpath;\n    }\n  }\n\n  const parseLinkResult = parseLink(link.original);\n  let alias: string | undefined;\n\n  if (isWikilink && parseLinkResult?.alias) {\n    alias = parseLinkResult.alias;\n    shouldKeepAlias = true;\n  }\n\n  alias ??= shouldResetAlias(normalizeOptionalProperties<ShouldResetAliasOptions>({\n      app,\n      displayText: link.displayText,\n      isWikilink,\n      newSourcePathOrFile,\n      oldSourcePathOrFile,\n      oldTargetPath,\n      targetPathOrFile: targetFile\n    }))\n    ? undefined\n    : parseLinkResult?.alias;\n\n  if (!shouldKeepAlias) {\n    if (alias === basename(oldTargetPath, extname(oldTargetPath))) {\n      alias = targetFile.basename;\n    } else if (alias === basename(oldTargetPath)) {\n      alias = targetFile.name;\n    }\n  }\n\n  const newLink = generateMarkdownLink(normalizeOptionalProperties<GenerateMarkdownLinkOptions>({\n    alias,\n    app,\n    linkStyle,\n    originalLink: link.original,\n    sourcePathOrFile: newSourcePathOrFile,\n    subpath,\n    targetPathOrFile: targetFile\n  }));\n  return newLink;\n}\n\n/**\n * Updates the links in a content string based on the provided parameters.\n *\n * @param options - The options for updating the links.\n * @returns A {@link Promise} that resolves to the content with updated links.\n */\nexport async function updateLinksInContent(options: UpdateLinksInContentOptions): Promise<string> {\n  const {\n    app,\n    content,\n    linkStyle,\n    newSourcePathOrFile,\n    oldSourcePathOrFile,\n    shouldUpdateEmbedOnlyLinks,\n    shouldUpdateFileNameAlias\n  } = options;\n\n  return await editLinksInContent(app, content, (link) => {\n    const isEmbedLink = testEmbed(link.original);\n    if (shouldUpdateEmbedOnlyLinks !== undefined && shouldUpdateEmbedOnlyLinks !== isEmbedLink) {\n      return;\n    }\n    return convertLink(normalizeOptionalProperties<ConvertLinkOptions>({\n      app,\n      link,\n      linkStyle,\n      newSourcePathOrFile,\n      oldSourcePathOrFile,\n      shouldUpdateFileNameAlias\n    }));\n  });\n}\n\n/**\n * Updates the links in a file based on the provided parameters.\n *\n * @param options - The options for updating the links.\n * @returns A {@link Promise} that resolves when the links are updated.\n */\nexport async function updateLinksInFile(options: UpdateLinksInFileOptions): Promise<void> {\n  const {\n    app,\n    linkStyle,\n    newSourcePathOrFile,\n    oldSourcePathOrFile,\n    shouldUpdateEmbedOnlyLinks,\n    shouldUpdateFileNameAlias\n  } = options;\n\n  if (isCanvasFile(app, newSourcePathOrFile) && !app.internalPlugins.getEnabledPluginById(InternalPluginName.Canvas)) {\n    return;\n  }\n\n  await editLinks(app, newSourcePathOrFile, (link) => {\n    const isEmbedLink = testEmbed(link.original);\n    if (shouldUpdateEmbedOnlyLinks !== undefined && shouldUpdateEmbedOnlyLinks !== isEmbedLink) {\n      return;\n    }\n    return convertLink(normalizeOptionalProperties<ConvertLinkOptions>({\n      app,\n      link,\n      linkStyle,\n      newSourcePathOrFile,\n      oldSourcePathOrFile,\n      shouldUpdateFileNameAlias\n    }));\n  }, options);\n}\n\nfunction _fixFrontmatterMarkdownLinks(value: unknown, key: string, cache: CachedMetadata): boolean {\n  if (typeof value === 'string') {\n    const parseLinkResult = parseLink(value);\n    if (!parseLinkResult || parseLinkResult.isWikilink || parseLinkResult.isExternal) {\n      return false;\n    }\n\n    cache.frontmatterLinks ??= [];\n    let link = cache.frontmatterLinks.find((frontmatterLink) => frontmatterLink.key === key);\n\n    if (!link) {\n      link = {\n        key,\n        link: '',\n        original: ''\n      };\n      cache.frontmatterLinks.push(link);\n    }\n\n    link.link = parseLinkResult.url;\n    link.original = value;\n    if (parseLinkResult.alias !== undefined) {\n      link.displayText = parseLinkResult.alias;\n    }\n\n    return true;\n  }\n\n  if (typeof value !== 'object' || value === null) {\n    return false;\n  }\n\n  let hasFrontmatterLinks = false;\n\n  for (const [childKey, childValue] of Object.entries(value as GenericObject)) {\n    const hasChildFrontmatterLinks = _fixFrontmatterMarkdownLinks(childValue, key ? `${key}.${childKey}` : childKey, cache);\n    hasFrontmatterLinks ||= hasChildFrontmatterLinks;\n  }\n\n  return hasFrontmatterLinks;\n}\n\nfunction extractTextLinks(str: string, startOffset: number, endOffset: number, textLinks: ParseLinkResult[]): void {\n  if (startOffset > endOffset) {\n    return;\n  }\n\n  const textPart = str.slice(startOffset, endOffset + 1);\n  replaceAll(textPart, /(?<Url>\\S+)/g, (args, url) => {\n    if (!isUrl(url)) {\n      return;\n    }\n\n    textLinks.push({\n      encodedUrl: encodeUrl(url),\n      endOffset: startOffset + args.offset + url.length,\n      hasAngleBrackets: false,\n      isEmbed: false,\n      isExternal: true,\n      isWikilink: false,\n      raw: url,\n      startOffset: startOffset + args.offset,\n      url\n    });\n  });\n}\n\nfunction generateLinkText(app: App, targetFile: TFile, sourcePath: string, subpath: string, config: LinkConfig): string {\n  if (sourcePath === '/') {\n    sourcePath = '';\n  }\n\n  let linkText: string;\n\n  if (targetFile.path === sourcePath && subpath) {\n    linkText = subpath;\n  } else if (config.shouldForceRelativePath) {\n    linkText = relative(dirname(sourcePath), config.isWikilink ? trimMarkdownExtension(app, targetFile) : targetFile.path) + subpath;\n  } else {\n    linkText = app.metadataCache.fileToLinktext(targetFile, sourcePath, config.isWikilink) + subpath;\n  }\n\n  if (config.shouldForceRelativePath && config.shouldUseLeadingDot && !linkText.startsWith('.') && !linkText.startsWith('#')) {\n    linkText = `./${linkText}`;\n  }\n\n  return linkText;\n}\n\nfunction generateMarkdownLinkImpl(options: GenerateMarkdownLinkOptions): string {\n  const { app } = options;\n  const targetFile = getFile(app, options.targetPathOrFile, options.isNonExistingFileAllowed);\n  const sourcePath = getPath(app, options.sourcePathOrFile);\n  const subpath = options.subpath ?? '';\n\n  const linkConfig = getLinkConfig(options, targetFile);\n  const linkText = generateLinkText(app, targetFile, sourcePath, subpath, linkConfig);\n\n  return linkConfig.isWikilink\n    ? generateWikiLink(linkText, options.alias, linkConfig.isEmbed)\n    : generateMarkdownStyleLink(linkText, targetFile, options, linkConfig);\n}\n\nfunction generateMarkdownStyleLink(linkText: string, targetFile: TFile, options: GenerateMarkdownLinkOptions, config: LinkConfig): string {\n  const { app } = options;\n  const embedPrefix = config.isEmbed ? '!' : '';\n\n  const processedLinkText = config.shouldUseAngleBrackets\n    ? `<${linkText}>`\n    : encodeUrl(linkText);\n\n  let alias = options.alias ?? '';\n  if (!alias && (!config.isEmbed || !options.isEmptyEmbedAliasAllowed)) {\n    alias = !options.shouldIncludeAttachmentExtensionToEmbedAlias || isMarkdownFile(app, targetFile)\n      ? targetFile.basename\n      : targetFile.name;\n  }\n\n  const escapedAlias = replaceAll(alias, SPECIAL_MARKDOWN_LINK_SYMBOLS_REGEX, '\\\\$&');\n  return `${embedPrefix}[${escapedAlias}](${processedLinkText})`;\n}\n\nfunction generateWikiLink(linkText: string, alias: string | undefined, isEmbed: boolean): string {\n  const embedPrefix = isEmbed ? '!' : '';\n  const normalizedAlias = alias ?? '';\n\n  if (normalizedAlias && normalizedAlias.toLowerCase() === linkText.toLowerCase()) {\n    return `${embedPrefix}[[${normalizedAlias}]]`;\n  }\n\n  const aliasPart = normalizedAlias ? `|${normalizedAlias}` : '';\n  return `${embedPrefix}[[${linkText}${aliasPart}]]`;\n}\n\nasync function getFileChanges(\n  cache: CachedMetadata | null,\n  isCanvasFileCache: boolean,\n  linkConverter: (link: Reference, abortSignal: AbortSignal) => Promisable<MaybeReturn<string>>,\n  abortSignal?: AbortSignal\n): Promise<FileChange[]> {\n  abortSignal ??= abortSignalNever();\n  abortSignal.throwIfAborted();\n\n  if (!cache) {\n    return [];\n  }\n\n  const changes: FileChange[] = [];\n\n  const tablePositions: TablePosition[] = (cache.sections ?? []).filter((section) => section.type === 'table').map((section) => ({\n    end: section.position.end.offset,\n    start: section.position.start.offset\n  }));\n\n  for (const link of getAllLinks(cache)) {\n    abortSignal.throwIfAborted();\n    const newContent = await linkConverter(link, abortSignal);\n    abortSignal.throwIfAborted();\n    if (newContent === undefined) {\n      continue;\n    }\n\n    const fileChange = referenceToFileChange(link, newContent);\n\n    if (isCanvasFileCache) {\n      if (isCanvasChange(fileChange)) {\n        changes.push(fileChange);\n      } else {\n        console.warn('Unsupported file change', fileChange);\n      }\n    } else {\n      if (shouldEscapeWikilinkDivider(fileChange, tablePositions)) {\n        fileChange.newContent = fileChange.newContent.replaceAll(UNESCAPED_WIKILINK_DIVIDER_REGEXP, ESCAPED_WIKILINK_DIVIDER);\n      }\n\n      changes.push(fileChange);\n    }\n  }\n  return changes;\n}\n\nfunction getLinkConfig(options: GenerateMarkdownLinkOptions, targetFile: TFile): LinkConfig {\n  const { app } = options;\n  return {\n    isEmbed: options.isEmbed ?? (options.originalLink ? testEmbed(options.originalLink) : undefined) ?? !isMarkdownFile(app, targetFile),\n    isWikilink: shouldUseWikilinkStyle(app, options.originalLink, options.linkStyle),\n    shouldForceRelativePath: options.shouldForceRelativePath ?? shouldUseRelativeLinks(app),\n    shouldUseAngleBrackets: options.shouldUseAngleBrackets ?? (options.originalLink ? testAngleBrackets(options.originalLink) : undefined) ?? false,\n    shouldUseLeadingDot: options.shouldUseLeadingDot ?? (options.originalLink ? testLeadingDot(options.originalLink) : undefined) ?? false\n  };\n}\n\nfunction getRawLink(node: Node, str: string): string {\n  return str.slice(node.position?.start.offset ?? 0, node.position?.end.offset ?? 0);\n}\n\nfunction parseLinkNode(node: Link, str: string): ParseLinkResult {\n  const OPEN_ANGLE_BRACKET = '<';\n  const LINK_ALIAS_SUFFIX = '](';\n  const LINK_SUFFIX = ')';\n  const raw = getRawLink(node, str);\n  const aliasNode = node.children[0] as Text | undefined;\n  const rawUrl = str.slice((aliasNode?.position?.end.offset ?? 1) + LINK_ALIAS_SUFFIX.length, (node.position?.end.offset ?? 0) - LINK_SUFFIX.length);\n  const hasAngleBrackets = raw.startsWith(OPEN_ANGLE_BRACKET) || rawUrl.startsWith(OPEN_ANGLE_BRACKET);\n  const isExternal = isUrl(node.url);\n  let url = node.url;\n  if (!isExternal && !hasAngleBrackets) {\n    try {\n      url = decodeURIComponent(url);\n    } catch (error) {\n      console.error(`Failed to decode URL ${url}`, error);\n    }\n  }\n  return normalizeOptionalProperties<ParseLinkResult>({\n    alias: aliasNode?.value,\n    encodedUrl: isExternal ? encodeUrl(url) : undefined,\n    endOffset: node.position?.end.offset ?? 0,\n    hasAngleBrackets,\n    isEmbed: false,\n    isExternal,\n    isWikilink: false,\n    raw,\n    startOffset: node.position?.start.offset ?? 0,\n    title: node.title ?? undefined,\n    url\n  });\n}\n\nfunction parseWikilinkNode(node: WikiLinkNode, str: string): ParseLinkResult {\n  return normalizeOptionalProperties<ParseLinkResult>({\n    alias: str.includes(WIKILINK_DIVIDER) ? node.data.alias : undefined,\n    endOffset: node.position?.end.offset ?? 0,\n    isEmbed: false,\n    isExternal: false,\n    isWikilink: true,\n    raw: getRawLink(node, str),\n    startOffset: node.position?.start.offset ?? 0,\n    url: node.value\n  });\n}\n\nfunction shouldEscapeWikilinkDivider(fileChange: FileChange, tablePositions: TablePosition[]): boolean {\n  if (!isContentChange(fileChange)) {\n    return false;\n  }\n\n  if (!UNESCAPED_WIKILINK_DIVIDER_REGEXP.test(fileChange.newContent)) {\n    return false;\n  }\n\n  return tablePositions.some((tablePosition) =>\n    tablePosition.start <= fileChange.reference.position.start.offset && fileChange.reference.position.end.offset <= tablePosition.end\n  );\n}\n\nfunction shouldUseWikilinkStyle(app: App, originalLink?: string, linkStyle?: LinkStyle): boolean {\n  switch (linkStyle ?? LinkStyle.PreserveExisting) {\n    case LinkStyle.Markdown:\n      return false;\n    case LinkStyle.ObsidianSettingsDefault:\n      return shouldUseWikilinks(app);\n    case LinkStyle.PreserveExisting:\n      return originalLink === undefined ? shouldUseWikilinks(app) : testWikilink(originalLink);\n    case LinkStyle.Wikilink:\n      return true;\n    default:\n      throw new Error(`Invalid link style: ${linkStyle as string}.`);\n  }\n}\n"],
  "mappings": "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",
  "names": ["LinkStyle", "remarkParse"]
}

|
|
647
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Link.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utilities for handling and updating links within Obsidian vaults. It includes\n * functions to split paths, update links in files, and generate markdown links with various options.\n */\n\nimport type {\n  Link,\n  Text\n} from 'mdast';\nimport type {\n  App,\n  CachedMetadata,\n  Reference,\n  TFile\n} from 'obsidian';\nimport type { Promisable } from 'type-fest';\nimport type { Node } from 'unist';\n\nimport {\n  normalizePath,\n  parseLinktext\n} from 'obsidian';\nimport { InternalPluginName } from 'obsidian-typings/implementations';\nimport { remark } from 'remark';\nimport remarkParse from 'remark-parse';\nimport { wikiLinkPlugin } from 'remark-wiki-link';\nimport { visit } from 'unist-util-visit';\n\nimport type { GenericObject } from '../ObjectUtils.ts';\nimport type { MaybeReturn } from '../Type.ts';\nimport type { FileChange } from './FileChange.ts';\nimport type { PathOrFile } from './FileSystem.ts';\nimport type { ProcessOptions } from './Vault.ts';\n\nimport { abortSignalNever } from '../AbortController.ts';\nimport {\n  normalizeOptionalProperties,\n  toJson\n} from '../ObjectUtils.ts';\nimport {\n  basename,\n  dirname,\n  extname,\n  join,\n  relative\n} from '../Path.ts';\nimport {\n  normalize,\n  replaceAll\n} from '../String.ts';\nimport { isUrl } from '../url.ts';\nimport {\n  applyContentChanges,\n  applyFileChanges,\n  isCanvasChange,\n  isContentChange\n} from './FileChange.ts';\nimport {\n  getFile,\n  getPath,\n  isCanvasFile,\n  isMarkdownFile,\n  trimMarkdownExtension\n} from './FileSystem.ts';\nimport {\n  getAllLinks,\n  getBacklinksForFileSafe,\n  getCacheSafe,\n  parseMetadata,\n  tempRegisterFilesAndRun\n} from './MetadataCache.ts';\nimport {\n  shouldUseRelativeLinks,\n  shouldUseWikilinks\n} from './ObsidianSettings.ts';\nimport {\n  isCanvasFileNodeReference,\n  referenceToFileChange\n} from './Reference.ts';\n\nconst ESCAPED_WIKILINK_DIVIDER = '\\\\|';\n\n/**\n * Regular expression for special link symbols.\n */\n// eslint-disable-next-line no-control-regex\nconst SPECIAL_LINK_SYMBOLS_REGEXP = /[\\\\\\x00\\x08\\x0B\\x0C\\x0E-\\x1F ]/g;\n\n/**\n * Regular expression for special markdown link symbols.\n */\nconst SPECIAL_MARKDOWN_LINK_SYMBOLS_REGEX = /[\\\\[\\]<>_*~=`$]/g;\n\n/**\n * Regular expression for unescaped pipes.\n */\nconst UNESCAPED_WIKILINK_DIVIDER_REGEXP = /(?<!\\\\)\\|/g;\n\nconst WIKILINK_DIVIDER = '|';\n\n/**\n * A style of the link.\n */\nexport enum LinkStyle {\n  /**\n   * Force the link to be in markdown format.\n   */\n  Markdown = 'Markdown',\n\n  /**\n   * Use the default link style defined in Obsidian settings.\n   */\n  ObsidianSettingsDefault = 'ObsidianSettingsDefault',\n\n  /**\n   * Preserve the existing link style.\n   */\n  PreserveExisting = 'PreserveExisting',\n\n  /**\n   * Force the link to be in wikilink format.\n   */\n  Wikilink = 'Wikilink'\n}\n\n/**\n * Options for {@link convertLink}.\n */\nexport interface ConvertLinkOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A reference for the link.\n   */\n  link: Reference;\n\n  /**\n   * A style of the link.\n   */\n  linkStyle?: LinkStyle;\n\n  /**\n   * A source file containing the link.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * An old path of the link.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * Whether to update file name alias. Defaults to `true`.\n   */\n  shouldUpdateFileNameAlias?: boolean;\n}\n\n/**\n * Wrapper for default options for {@link GenerateMarkdownLinkOptions}.\n */\nexport interface GenerateMarkdownLinkDefaultOptionsWrapper {\n  /**\n   * A default options for generating markdown links.\n   *\n   * @returns A default options for generating markdown links.\n   */\n  defaultOptionsFn(): Partial<GenerateMarkdownLinkOptions>;\n}\n\n/**\n * Options for {@link generateMarkdownLink}.\n */\nexport interface GenerateMarkdownLinkOptions {\n  /**\n   * An alias for the link.\n   */\n  alias?: string;\n\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * Indicates if the link should be embedded. If not provided, it will be inferred based on the file type.\n   */\n  isEmbed?: boolean;\n\n  /**\n   * Whether to allow an empty alias for embeds. Defaults to `true`.\n   */\n  isEmptyEmbedAliasAllowed?: boolean;\n\n  /**\n   * Whether to allow non-existing files. If `false` and `pathOrFile` is a non-existing file, an error will be thrown. Defaults to `false`.\n   */\n  isNonExistingFileAllowed?: boolean;\n\n  /**\n   * A style of the link.\n   */\n  linkStyle?: LinkStyle;\n\n  /**\n   * An original link text. If provided, it will be used to infer the values of `isEmbed`, `isWikilink`, `useLeadingDot`, and `useAngleBrackets`.\n   * These inferred values will be overridden by corresponding settings if specified.\n   */\n  originalLink?: string;\n\n  /**\n   * Indicates if the link should be relative. If not provided or `false`, it will be inferred based on the Obsidian settings.\n   */\n  shouldForceRelativePath?: boolean;\n\n  /**\n   * Whether to include the attachment extension in the embed alias. Has no effect if `allowEmptyEmbedAlias` is `true`. Defaults to `false`.\n   */\n  shouldIncludeAttachmentExtensionToEmbedAlias?: boolean;\n\n  /**\n   * Indicates if the link should use angle brackets. Defaults to `false`. Has no effect if `isWikilink` is `true`\n   */\n  shouldUseAngleBrackets?: boolean;\n\n  /**\n   * Indicates if the link should use a leading dot. Defaults to `false`. Has no effect if `isWikilink` is `true` or `isRelative` is `false`.\n   */\n  shouldUseLeadingDot?: boolean;\n\n  /**\n   * A source path of the link.\n   */\n  sourcePathOrFile: PathOrFile;\n\n  /**\n   * A subpath of the link.\n   */\n  subpath?: string;\n\n  /**\n   * A target path or file.\n   */\n  targetPathOrFile: PathOrFile;\n}\n\n/**\n * A result of parsing a link.\n */\nexport interface ParseLinkResult {\n  /**\n   * An alias of the link.\n   */\n  alias?: string;\n\n  /**\n   * An encoded URL of the link.\n   */\n  encodedUrl?: string;\n\n  /**\n   * An end offset of the link in the original text.\n   */\n  endOffset: number;\n\n  /**\n   * Indicates if the link has angle brackets.\n   */\n  hasAngleBrackets?: boolean;\n\n  /**\n   * Indicates if the link is an embed link.\n   */\n  isEmbed: boolean;\n\n  /**\n   * Indicates if the link is external.\n   */\n  isExternal: boolean;\n\n  /**\n   * Indicates if the link is a wikilink.\n   */\n  isWikilink: boolean;\n\n  /**\n   * A raw link text.\n   */\n  raw: string;\n\n  /**\n   * A start offset of the link in the original text.\n   */\n  startOffset: number;\n\n  /**\n   * A title of the link.\n   */\n  title?: string;\n\n  /**\n   * An URL of the link.\n   */\n  url: string;\n}\n\n/**\n * Options for {@link shouldResetAlias}.\n */\nexport interface ShouldResetAliasOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A display text of the link.\n   */\n  displayText: string | undefined;\n\n  /**\n   * Indicates if the link is a wikilink.\n   */\n  isWikilink?: boolean;\n\n  /**\n   * A source path of the link.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * An old source file containing the link.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * An old target path of the link.\n   */\n  oldTargetPath: PathOrFile;\n\n  /**\n   * A target path or file.\n   */\n  targetPathOrFile: PathOrFile;\n}\n\n/**\n * Splits a link into its link path and subpath.\n */\nexport interface SplitSubpathResult {\n  /**\n   * A link path.\n   */\n  linkPath: string;\n\n  /**\n   * A subpath.\n   */\n  subpath: string;\n}\n\n/**\n * Options for {@link updateLink}.\n */\nexport interface UpdateLinkOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A reference for the link.\n   */\n  link: Reference;\n\n  /**\n   * Whether to force markdown links.\n   */\n  linkStyle?: LinkStyle;\n\n  /**\n   * A source file containing the link.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * A file associated with the link.\n   */\n  newTargetPathOrFile: PathOrFile;\n\n  /**\n   * An old source file containing the link.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * An old path of the file.\n   */\n  oldTargetPathOrFile?: PathOrFile;\n\n  /**\n   * Whether to update file name alias. Defaults to `true`.\n   */\n  shouldUpdateFileNameAlias?: boolean;\n}\n\n/**\n * Options for {@link updateLinksInFile}.\n */\nexport interface UpdateLinksInFileOptions extends ProcessOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A style of the link.\n   */\n  linkStyle?: LinkStyle;\n\n  /**\n   * A file to update the links in.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * An old path of the file.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * Whether to update only embedded links.\n   */\n  shouldUpdateEmbedOnlyLinks?: boolean;\n\n  /**\n   * Whether to update file name alias. Defaults to `true`.\n   */\n  shouldUpdateFileNameAlias?: boolean;\n}\n\ninterface LinkConfig {\n  isEmbed: boolean;\n  isWikilink: boolean;\n  shouldForceRelativePath: boolean;\n  shouldUseAngleBrackets: boolean;\n  shouldUseLeadingDot: boolean;\n}\n\ninterface TablePosition {\n  end: number;\n  start: number;\n}\n\n/**\n * Options for {@link updateLinksInContent}.\n */\ninterface UpdateLinksInContentOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A content to update the links in.\n   */\n  content: string;\n\n  /**\n   * A style of the link.\n   */\n  linkStyle?: LinkStyle;\n\n  /**\n   * A new source path or file.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * An old source path or file.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * Whether to update only embedded links.\n   */\n  shouldUpdateEmbedOnlyLinks?: boolean;\n\n  /**\n   * Whether to update file name alias.\n   */\n  shouldUpdateFileNameAlias?: boolean;\n}\n\ninterface WikiLinkNode extends Node {\n  data: {\n    alias: string;\n  };\n  value: string;\n}\n\n/**\n * Converts a link to a new path.\n *\n * @param options - The options for converting the link.\n * @returns The converted link.\n */\nexport function convertLink(options: ConvertLinkOptions): string {\n  const targetFile = extractLinkFile(options.app, options.link, options.oldSourcePathOrFile ?? options.newSourcePathOrFile);\n  if (!targetFile) {\n    return options.link.original;\n  }\n\n  return updateLink(normalizeOptionalProperties<UpdateLinkOptions>({\n    app: options.app,\n    link: options.link,\n    linkStyle: options.linkStyle,\n    newSourcePathOrFile: options.newSourcePathOrFile,\n    newTargetPathOrFile: targetFile,\n    oldSourcePathOrFile: options.oldSourcePathOrFile,\n    shouldUpdateFileNameAlias: options.shouldUpdateFileNameAlias\n  }));\n}\n\n/**\n * Edits the backlinks for a file or path.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or file to edit the backlinks for.\n * @param linkConverter - The function that converts each link.\n * @param processOptions - Optional options for retrying the operation.\n * @returns A {@link Promise} that resolves when the backlinks have been edited.\n */\nexport async function editBacklinks(\n  app: App,\n  pathOrFile: PathOrFile,\n  linkConverter: (link: Reference) => Promisable<MaybeReturn<string>>,\n  processOptions: ProcessOptions = {}\n): Promise<void> {\n  const backlinks = await getBacklinksForFileSafe(app, pathOrFile, processOptions);\n  for (const backlinkNotePath of backlinks.keys()) {\n    const currentLinks = backlinks.get(backlinkNotePath) ?? [];\n    const linkJsons = new Set<string>(currentLinks.map((link) => toJson(link)));\n    await editLinks(app, backlinkNotePath, (link) => {\n      const linkJson = toJson(link);\n      if (!linkJsons.has(linkJson)) {\n        return;\n      }\n\n      return linkConverter(link);\n    }, processOptions);\n  }\n}\n\n/**\n * Edits the backlinks for a file or path.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or file to edit the backlinks for.\n * @param linkConverter - The function that converts each link.\n * @param processOptions - Optional options for retrying the operation.\n * @returns A {@link Promise} that resolves when the backlinks have been edited.\n */\nexport async function editLinks(\n  app: App,\n  pathOrFile: PathOrFile,\n  linkConverter: (link: Reference) => Promisable<MaybeReturn<string>>,\n  processOptions: ProcessOptions = {}\n): Promise<void> {\n  await applyFileChanges(app, pathOrFile, async (abortSignal, content) => {\n    const cache = await getCacheSafe(app, pathOrFile);\n    abortSignal.throwIfAborted();\n    const file = getFile(app, pathOrFile);\n    const cachedContent = await app.vault.cachedRead(file);\n    abortSignal.throwIfAborted();\n    if (content !== cachedContent) {\n      return null;\n    }\n\n    return await getFileChanges(cache, isCanvasFile(app, pathOrFile), linkConverter, abortSignal);\n  }, processOptions);\n}\n\n/**\n * Edits the links in a content string.\n *\n * @param app - The Obsidian application instance.\n * @param content - The content to edit the links in.\n * @param linkConverter - The function that converts each link.\n * @param abortSignal - The abort signal to control the execution of the function.\n * @returns The promise that resolves to the updated content.\n */\nexport async function editLinksInContent(\n  app: App,\n  content: string,\n  linkConverter: (link: Reference) => Promisable<MaybeReturn<string>>,\n  abortSignal?: AbortSignal\n): Promise<string> {\n  abortSignal ??= abortSignalNever();\n  abortSignal.throwIfAborted();\n\n  const newContent = await applyContentChanges(abortSignal, content, '', async () => {\n    const cache = await parseMetadata(app, content);\n    abortSignal.throwIfAborted();\n    const changes = await getFileChanges(cache, false, linkConverter, abortSignal);\n    abortSignal.throwIfAborted();\n    return changes;\n  });\n  abortSignal.throwIfAborted();\n\n  if (newContent === null) {\n    throw new Error('Failed to update links in content');\n  }\n\n  return newContent;\n}\n\n/**\n * Encodes a URL.\n *\n * @param url - The URL to encode.\n * @returns The encoded URL.\n */\nexport function encodeUrl(url: string): string {\n  return replaceAll(url, SPECIAL_LINK_SYMBOLS_REGEXP, ({ substring: specialLinkSymbol }) => encodeURIComponent(specialLinkSymbol));\n}\n\n/**\n * Extracts the file associated with a link.\n *\n * @param app - The Obsidian application instance.\n * @param link - The reference cache for the link.\n * @param sourcePathOrFile - The source path or file.\n * @param shouldAllowNonExistingFile - Whether to allow non-existing files. Defaults to `false`.\n * @returns The file associated with the link, or null if not found.\n */\nexport function extractLinkFile(app: App, link: Reference, sourcePathOrFile: PathOrFile, shouldAllowNonExistingFile = false): null | TFile {\n  const { linkPath } = splitSubpath(link.link);\n  const sourcePath = getPath(app, sourcePathOrFile);\n  const file = app.metadataCache.getFirstLinkpathDest(linkPath, sourcePath);\n  if (file) {\n    return file;\n  }\n\n  if (!shouldAllowNonExistingFile) {\n    return null;\n  }\n\n  if (linkPath.startsWith('/')) {\n    return getFile(app, linkPath, true);\n  }\n\n  const fullLinkPath = join(dirname(sourcePath), `./${linkPath}`);\n\n  if (fullLinkPath.startsWith('../')) {\n    return null;\n  }\n\n  return getFile(app, fullLinkPath, true);\n}\n\n/**\n * Fixes the frontmatter markdown links in the provided metadata cache.\n *\n * @param cache - The metadata cache to fix the frontmatter markdown links in.\n * @returns Whether the frontmatter markdown links were fixed.\n */\nexport function fixFrontmatterMarkdownLinks(cache: CachedMetadata): boolean {\n  return _fixFrontmatterMarkdownLinks(cache.frontmatter, '', cache);\n}\n\n/**\n * Generates a markdown link based on the provided parameters.\n *\n * @param options - The options for generating the markdown link.\n * @returns The generated markdown link.\n */\nexport function generateMarkdownLink(options: GenerateMarkdownLinkOptions): string {\n  const { app } = options;\n\n  const configurableDefaultOptionsFn = (app.fileManager.generateMarkdownLink as Partial<GenerateMarkdownLinkDefaultOptionsWrapper>).defaultOptionsFn\n    ?? ((): Partial<GenerateMarkdownLinkOptions> => ({}));\n  const configurableDefaultOptions = configurableDefaultOptionsFn();\n\n  const DEFAULT_OPTIONS: Partial<GenerateMarkdownLinkOptions> = {\n    isEmptyEmbedAliasAllowed: true\n  };\n\n  options = { ...DEFAULT_OPTIONS, ...configurableDefaultOptions, ...options };\n\n  const targetFile = getFile(app, options.targetPathOrFile, options.isNonExistingFileAllowed);\n\n  return tempRegisterFilesAndRun(app, [targetFile], () => generateMarkdownLinkImpl(options));\n}\n\n/**\n * Parses a link into its components.\n *\n * @param str - The link to parse.\n * @returns The parsed link.\n */\nexport function parseLink(str: string): null | ParseLinkResult {\n  const links = parseLinks(str);\n  return links[0]?.raw === str ? links[0] : null;\n}\n\n/**\n * Parses all links in a string.\n *\n * @param str - The string to parse the links in.\n * @returns The parsed links.\n */\nexport function parseLinks(str: string): ParseLinkResult[] {\n  const embedSymbolOffsets = new Set<number>();\n\n  const EMBED_LINK_PREFIX = '![';\n  const NO_EMBED_LINK_PREFIX = ' [';\n\n  const noEmbedStr = replaceAll(str, EMBED_LINK_PREFIX, (args) => {\n    embedSymbolOffsets.add(args.offset);\n    return NO_EMBED_LINK_PREFIX;\n  });\n\n  const processor = remark().use(remarkParse).use(wikiLinkPlugin, { aliasDivider: WIKILINK_DIVIDER });\n  const root = processor.parse(noEmbedStr);\n\n  const links: ParseLinkResult[] = [];\n  const textLinks: ParseLinkResult[] = [];\n\n  visit(root, (node: Node) => {\n    let link: ParseLinkResult;\n    switch (node.type) {\n      case 'link':\n        link = parseLinkNode(node as Link, str);\n        break;\n      case 'wikiLink':\n        link = parseWikilinkNode(node as WikiLinkNode, str);\n        break;\n      default:\n        return;\n    }\n\n    if (embedSymbolOffsets.has(link.startOffset - 1)) {\n      link.isEmbed = true;\n      link.startOffset--;\n      link.raw = `!${link.raw}`;\n    }\n    links.push(link);\n  });\n\n  links.sort((a, b) => a.startOffset - b.startOffset);\n\n  let textStartOffset = 0;\n\n  for (const link of links) {\n    extractTextLinks(str, textStartOffset, link.startOffset - 1, textLinks);\n    textStartOffset = link.endOffset + 1;\n  }\n\n  extractTextLinks(str, textStartOffset, str.length - 1, textLinks);\n\n  links.push(...textLinks);\n  links.sort((a, b) => a.startOffset - b.startOffset);\n\n  return links;\n}\n\n/**\n * Determines if the alias of a link should be reset.\n *\n * @param options - The options for determining if the alias should be reset.\n * @returns Whether the alias should be reset.\n */\nexport function shouldResetAlias(options: ShouldResetAliasOptions): boolean {\n  const {\n    app,\n    displayText,\n    isWikilink,\n    newSourcePathOrFile,\n    oldSourcePathOrFile,\n    oldTargetPath,\n    targetPathOrFile\n  } = options;\n  if (isWikilink === false) {\n    return false;\n  }\n\n  if (!displayText) {\n    return true;\n  }\n\n  const targetFile = getFile(app, targetPathOrFile, true);\n  const newSourcePath = getPath(app, newSourcePathOrFile);\n  const oldSourcePath = getPath(app, oldSourcePathOrFile ?? newSourcePathOrFile);\n  const newSourceFolder = dirname(newSourcePath);\n  const oldSourceFolder = dirname(oldSourcePath);\n  const aliasesToReset = new Set<string>();\n\n  for (const pathOrFile of [targetFile.path, oldTargetPath]) {\n    if (!pathOrFile) {\n      continue;\n    }\n\n    const path = getPath(app, pathOrFile);\n    aliasesToReset.add(path);\n    aliasesToReset.add(basename(path));\n    aliasesToReset.add(relative(newSourceFolder, path));\n    aliasesToReset.add(relative(oldSourceFolder, path));\n  }\n\n  for (const sourcePath of [oldSourcePath, newSourcePath]) {\n    aliasesToReset.add(app.metadataCache.fileToLinktext(targetFile, sourcePath, false));\n  }\n\n  const cleanDisplayText = replaceAll(normalizePath(displayText.split(' > ')[0] ?? ''), /^\\.\\//g, '').toLowerCase();\n\n  for (const alias of aliasesToReset) {\n    if (alias.toLowerCase() === cleanDisplayText) {\n      return true;\n    }\n\n    const folder = dirname(alias);\n    const base = basename(alias, extname(alias));\n    if (join(folder, base).toLowerCase() === cleanDisplayText) {\n      return true;\n    }\n  }\n\n  return false;\n}\n\n/**\n * Splits a link into its link path and subpath.\n *\n * @param link - The link to split.\n * @returns An object containing the link path and subpath.\n */\nexport function splitSubpath(link: string): SplitSubpathResult {\n  const parsed = parseLinktext(normalize(link));\n  return {\n    linkPath: parsed.path,\n    subpath: parsed.subpath\n  };\n}\n\n/**\n * Tests whether a link uses angle brackets, possibly embed:\n * `[title](<link>)`, `![title](<link>)`.\n *\n * @param link - Link to test\n * @returns Whether the link uses angle brackets\n */\nexport function testAngleBrackets(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.hasAngleBrackets ?? false;\n}\n\n/**\n * Tests whether a link is an embed link:\n * `![[link]]`, `![title](link)`.\n *\n * @param link - Link to test\n * @returns Whether the link is an embed link\n */\nexport function testEmbed(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.isEmbed ?? false;\n}\n\n/**\n * Tests whether a link has a leading dot, possibly embed:\n * `[[./link]]`, `[title](./link)`, `[title](<./link>)`,\n * `![[./link]]`, `![title](./link)`, `![title](<./link>)`.\n *\n * @param link - Link to test\n * @returns Whether the link has a leading dot\n */\nexport function testLeadingDot(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.url.startsWith('./') ?? false;\n}\n\n/**\n * Tests whether a link is a wikilink, possibly embed:\n * `[[link]]`, `![[link]]`.\n *\n * @param link - Link to test\n * @returns Whether the link is a wikilink\n */\nexport function testWikilink(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.isWikilink ?? false;\n}\n\n/**\n * Updates a link based on the provided parameters.\n *\n * @param options - The options for updating the link.\n * @returns The updated link.\n */\nexport function updateLink(options: UpdateLinkOptions): string {\n  const {\n    app,\n    link,\n    linkStyle,\n    newSourcePathOrFile,\n    newTargetPathOrFile,\n    oldSourcePathOrFile,\n    oldTargetPathOrFile,\n    shouldUpdateFileNameAlias\n  } = options;\n  if (!newTargetPathOrFile) {\n    return link.original;\n  }\n  const targetFile = getFile(app, newTargetPathOrFile, true);\n  const oldTargetPath = getPath(app, oldTargetPathOrFile ?? newTargetPathOrFile);\n  const isWikilink = shouldUseWikilinkStyle(app, link.original, linkStyle);\n\n  const { subpath } = splitSubpath(link.link);\n  let shouldKeepAlias = !shouldUpdateFileNameAlias;\n\n  if (isCanvasFile(app, newSourcePathOrFile)) {\n    if (isCanvasFileNodeReference(link)) {\n      return targetFile.path + subpath;\n    }\n  }\n\n  const parseLinkResult = parseLink(link.original);\n  let alias: string | undefined;\n\n  if (isWikilink && parseLinkResult?.alias) {\n    alias = parseLinkResult.alias;\n    shouldKeepAlias = true;\n  }\n\n  alias ??= shouldResetAlias(normalizeOptionalProperties<ShouldResetAliasOptions>({\n      app,\n      displayText: link.displayText,\n      isWikilink,\n      newSourcePathOrFile,\n      oldSourcePathOrFile,\n      oldTargetPath,\n      targetPathOrFile: targetFile\n    }))\n    ? undefined\n    : parseLinkResult?.alias;\n\n  if (!shouldKeepAlias) {\n    if (alias === basename(oldTargetPath, extname(oldTargetPath))) {\n      alias = targetFile.basename;\n    } else if (alias === basename(oldTargetPath)) {\n      alias = targetFile.name;\n    }\n  }\n\n  const newLink = generateMarkdownLink(normalizeOptionalProperties<GenerateMarkdownLinkOptions>({\n    alias,\n    app,\n    linkStyle,\n    originalLink: link.original,\n    sourcePathOrFile: newSourcePathOrFile,\n    subpath,\n    targetPathOrFile: targetFile\n  }));\n  return newLink;\n}\n\n/**\n * Updates the links in a content string based on the provided parameters.\n *\n * @param options - The options for updating the links.\n * @returns A {@link Promise} that resolves to the content with updated links.\n */\nexport async function updateLinksInContent(options: UpdateLinksInContentOptions): Promise<string> {\n  const {\n    app,\n    content,\n    linkStyle,\n    newSourcePathOrFile,\n    oldSourcePathOrFile,\n    shouldUpdateEmbedOnlyLinks,\n    shouldUpdateFileNameAlias\n  } = options;\n\n  return await editLinksInContent(app, content, (link) => {\n    const isEmbedLink = testEmbed(link.original);\n    if (shouldUpdateEmbedOnlyLinks !== undefined && shouldUpdateEmbedOnlyLinks !== isEmbedLink) {\n      return;\n    }\n    return convertLink(normalizeOptionalProperties<ConvertLinkOptions>({\n      app,\n      link,\n      linkStyle,\n      newSourcePathOrFile,\n      oldSourcePathOrFile,\n      shouldUpdateFileNameAlias\n    }));\n  });\n}\n\n/**\n * Updates the links in a file based on the provided parameters.\n *\n * @param options - The options for updating the links.\n * @returns A {@link Promise} that resolves when the links are updated.\n */\nexport async function updateLinksInFile(options: UpdateLinksInFileOptions): Promise<void> {\n  const {\n    app,\n    linkStyle,\n    newSourcePathOrFile,\n    oldSourcePathOrFile,\n    shouldUpdateEmbedOnlyLinks,\n    shouldUpdateFileNameAlias\n  } = options;\n\n  if (isCanvasFile(app, newSourcePathOrFile) && !app.internalPlugins.getEnabledPluginById(InternalPluginName.Canvas)) {\n    return;\n  }\n\n  await editLinks(app, newSourcePathOrFile, (link) => {\n    const isEmbedLink = testEmbed(link.original);\n    if (shouldUpdateEmbedOnlyLinks !== undefined && shouldUpdateEmbedOnlyLinks !== isEmbedLink) {\n      return;\n    }\n    return convertLink(normalizeOptionalProperties<ConvertLinkOptions>({\n      app,\n      link,\n      linkStyle,\n      newSourcePathOrFile,\n      oldSourcePathOrFile,\n      shouldUpdateFileNameAlias\n    }));\n  }, options);\n}\n\nfunction _fixFrontmatterMarkdownLinks(value: unknown, key: string, cache: CachedMetadata): boolean {\n  if (typeof value === 'string') {\n    const parseLinkResult = parseLink(value);\n    if (!parseLinkResult || parseLinkResult.isWikilink || parseLinkResult.isExternal) {\n      return false;\n    }\n\n    cache.frontmatterLinks ??= [];\n    let link = cache.frontmatterLinks.find((frontmatterLink) => frontmatterLink.key === key);\n\n    if (!link) {\n      link = {\n        key,\n        link: '',\n        original: ''\n      };\n      cache.frontmatterLinks.push(link);\n    }\n\n    link.link = parseLinkResult.url;\n    link.original = value;\n    if (parseLinkResult.alias !== undefined) {\n      link.displayText = parseLinkResult.alias;\n    }\n\n    return true;\n  }\n\n  if (typeof value !== 'object' || value === null) {\n    return false;\n  }\n\n  let hasFrontmatterLinks = false;\n\n  for (const [childKey, childValue] of Object.entries(value as GenericObject)) {\n    const hasChildFrontmatterLinks = _fixFrontmatterMarkdownLinks(childValue, key ? `${key}.${childKey}` : childKey, cache);\n    hasFrontmatterLinks ||= hasChildFrontmatterLinks;\n  }\n\n  return hasFrontmatterLinks;\n}\n\nfunction extractTextLinks(str: string, startOffset: number, endOffset: number, textLinks: ParseLinkResult[]): void {\n  if (startOffset > endOffset) {\n    return;\n  }\n\n  const textPart = str.slice(startOffset, endOffset + 1);\n  replaceAll(textPart, /(?<Url>\\S+)/g, (args, url) => {\n    if (!isUrl(url)) {\n      return;\n    }\n\n    textLinks.push({\n      encodedUrl: encodeUrl(url),\n      endOffset: startOffset + args.offset + url.length,\n      hasAngleBrackets: false,\n      isEmbed: false,\n      isExternal: true,\n      isWikilink: false,\n      raw: url,\n      startOffset: startOffset + args.offset,\n      url\n    });\n  });\n}\n\nfunction generateLinkText(app: App, targetFile: TFile, sourcePath: string, subpath: string, config: LinkConfig): string {\n  if (sourcePath === '/') {\n    sourcePath = '';\n  }\n\n  let linkText: string;\n\n  if (targetFile.path === sourcePath && subpath) {\n    linkText = subpath;\n  } else if (config.shouldForceRelativePath) {\n    linkText = relative(dirname(sourcePath), config.isWikilink ? trimMarkdownExtension(app, targetFile) : targetFile.path) + subpath;\n  } else {\n    linkText = app.metadataCache.fileToLinktext(targetFile, sourcePath, config.isWikilink) + subpath;\n  }\n\n  if (config.shouldForceRelativePath && config.shouldUseLeadingDot && !linkText.startsWith('.') && !linkText.startsWith('#')) {\n    linkText = `./${linkText}`;\n  }\n\n  return linkText;\n}\n\nfunction generateMarkdownLinkImpl(options: GenerateMarkdownLinkOptions): string {\n  const { app } = options;\n  const targetFile = getFile(app, options.targetPathOrFile, options.isNonExistingFileAllowed);\n  const sourcePath = getPath(app, options.sourcePathOrFile);\n  const subpath = options.subpath ?? '';\n\n  const linkConfig = getLinkConfig(options, targetFile);\n  const linkText = generateLinkText(app, targetFile, sourcePath, subpath, linkConfig);\n\n  return linkConfig.isWikilink\n    ? generateWikiLink(linkText, options.alias, linkConfig.isEmbed)\n    : generateMarkdownStyleLink(linkText, targetFile, options, linkConfig);\n}\n\nfunction generateMarkdownStyleLink(linkText: string, targetFile: TFile, options: GenerateMarkdownLinkOptions, config: LinkConfig): string {\n  const { app } = options;\n  const embedPrefix = config.isEmbed ? '!' : '';\n\n  const processedLinkText = config.shouldUseAngleBrackets\n    ? `<${linkText}>`\n    : encodeUrl(linkText);\n\n  let alias = options.alias ?? '';\n  if (!alias && (!config.isEmbed || !options.isEmptyEmbedAliasAllowed)) {\n    alias = !options.shouldIncludeAttachmentExtensionToEmbedAlias || isMarkdownFile(app, targetFile)\n      ? targetFile.basename\n      : targetFile.name;\n  }\n\n  const escapedAlias = replaceAll(alias, SPECIAL_MARKDOWN_LINK_SYMBOLS_REGEX, '\\\\$&');\n  return `${embedPrefix}[${escapedAlias}](${processedLinkText})`;\n}\n\nfunction generateWikiLink(linkText: string, alias: string | undefined, isEmbed: boolean): string {\n  const embedPrefix = isEmbed ? '!' : '';\n  const normalizedAlias = alias ?? '';\n\n  if (normalizedAlias && normalizedAlias.toLowerCase() === linkText.toLowerCase()) {\n    return `${embedPrefix}[[${normalizedAlias}]]`;\n  }\n\n  const aliasPart = normalizedAlias ? `|${normalizedAlias}` : '';\n  return `${embedPrefix}[[${linkText}${aliasPart}]]`;\n}\n\nasync function getFileChanges(\n  cache: CachedMetadata | null,\n  isCanvasFileCache: boolean,\n  linkConverter: (link: Reference, abortSignal: AbortSignal) => Promisable<MaybeReturn<string>>,\n  abortSignal?: AbortSignal\n): Promise<FileChange[]> {\n  abortSignal ??= abortSignalNever();\n  abortSignal.throwIfAborted();\n\n  if (!cache) {\n    return [];\n  }\n\n  const changes: FileChange[] = [];\n\n  const tablePositions: TablePosition[] = (cache.sections ?? []).filter((section) => section.type === 'table').map((section) => ({\n    end: section.position.end.offset,\n    start: section.position.start.offset\n  }));\n\n  for (const link of getAllLinks(cache)) {\n    abortSignal.throwIfAborted();\n    const newContent = await linkConverter(link, abortSignal);\n    abortSignal.throwIfAborted();\n    if (newContent === undefined) {\n      continue;\n    }\n\n    const fileChange = referenceToFileChange(link, newContent);\n\n    if (isCanvasFileCache) {\n      if (isCanvasChange(fileChange)) {\n        changes.push(fileChange);\n      } else {\n        console.warn('Unsupported file change', fileChange);\n      }\n    } else {\n      if (shouldEscapeWikilinkDivider(fileChange, tablePositions)) {\n        fileChange.newContent = fileChange.newContent.replaceAll(UNESCAPED_WIKILINK_DIVIDER_REGEXP, ESCAPED_WIKILINK_DIVIDER);\n      }\n\n      changes.push(fileChange);\n    }\n  }\n  return changes;\n}\n\nfunction getLinkConfig(options: GenerateMarkdownLinkOptions, targetFile: TFile): LinkConfig {\n  const { app } = options;\n  return {\n    isEmbed: options.isEmbed ?? (options.originalLink ? testEmbed(options.originalLink) : undefined) ?? !isMarkdownFile(app, targetFile),\n    isWikilink: shouldUseWikilinkStyle(app, options.originalLink, options.linkStyle),\n    shouldForceRelativePath: options.shouldForceRelativePath ?? shouldUseRelativeLinks(app),\n    shouldUseAngleBrackets: options.shouldUseAngleBrackets ?? (options.originalLink ? testAngleBrackets(options.originalLink) : undefined) ?? false,\n    shouldUseLeadingDot: options.shouldUseLeadingDot ?? (options.originalLink ? testLeadingDot(options.originalLink) : undefined) ?? false\n  };\n}\n\nfunction getRawLink(node: Node, str: string): string {\n  return str.slice(node.position?.start.offset ?? 0, node.position?.end.offset ?? 0);\n}\n\nfunction parseLinkNode(node: Link, str: string): ParseLinkResult {\n  const OPEN_ANGLE_BRACKET = '<';\n  const LINK_ALIAS_SUFFIX = '](';\n  const LINK_SUFFIX = ')';\n  const raw = getRawLink(node, str);\n  const aliasNode = node.children[0] as Text | undefined;\n  const rawUrl = str.slice((aliasNode?.position?.end.offset ?? 1) + LINK_ALIAS_SUFFIX.length, (node.position?.end.offset ?? 0) - LINK_SUFFIX.length);\n  const hasAngleBrackets = raw.startsWith(OPEN_ANGLE_BRACKET) || rawUrl.startsWith(OPEN_ANGLE_BRACKET);\n  const isExternal = isUrl(node.url);\n  let url = node.url;\n  if (!isExternal && !hasAngleBrackets) {\n    try {\n      url = decodeURIComponent(url);\n    } catch (error) {\n      console.error(`Failed to decode URL ${url}`, error);\n    }\n  }\n  return normalizeOptionalProperties<ParseLinkResult>({\n    alias: aliasNode?.value,\n    encodedUrl: isExternal ? encodeUrl(url) : undefined,\n    endOffset: node.position?.end.offset ?? 0,\n    hasAngleBrackets,\n    isEmbed: false,\n    isExternal,\n    isWikilink: false,\n    raw,\n    startOffset: node.position?.start.offset ?? 0,\n    title: node.title ?? undefined,\n    url\n  });\n}\n\nfunction parseWikilinkNode(node: WikiLinkNode, str: string): ParseLinkResult {\n  return normalizeOptionalProperties<ParseLinkResult>({\n    alias: str.includes(WIKILINK_DIVIDER) ? node.data.alias : undefined,\n    endOffset: node.position?.end.offset ?? 0,\n    isEmbed: false,\n    isExternal: false,\n    isWikilink: true,\n    raw: getRawLink(node, str),\n    startOffset: node.position?.start.offset ?? 0,\n    url: node.value\n  });\n}\n\nfunction shouldEscapeWikilinkDivider(fileChange: FileChange, tablePositions: TablePosition[]): boolean {\n  if (!isContentChange(fileChange)) {\n    return false;\n  }\n\n  if (!UNESCAPED_WIKILINK_DIVIDER_REGEXP.test(fileChange.newContent)) {\n    return false;\n  }\n\n  return tablePositions.some((tablePosition) =>\n    tablePosition.start <= fileChange.reference.position.start.offset && fileChange.reference.position.end.offset <= tablePosition.end\n  );\n}\n\nfunction shouldUseWikilinkStyle(app: App, originalLink?: string, linkStyle?: LinkStyle): boolean {\n  switch (linkStyle ?? LinkStyle.PreserveExisting) {\n    case LinkStyle.Markdown:\n      return false;\n    case LinkStyle.ObsidianSettingsDefault:\n      return shouldUseWikilinks(app);\n    case LinkStyle.PreserveExisting:\n      return originalLink === undefined ? shouldUseWikilinks(app) : testWikilink(originalLink);\n    case LinkStyle.Wikilink:\n      return true;\n    default:\n      throw new Error(`Invalid link style: ${linkStyle as string}.`);\n  }\n}\n"],
  "mappings": "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",
  "names": ["LinkStyle", "remarkParse"]
}

|
package/dist/lib/esm/Async.mjs
CHANGED
|
@@ -213,46 +213,40 @@ async function retryWithTimeout(fn, retryOptions, stackTrace) {
|
|
|
213
213
|
);
|
|
214
214
|
}
|
|
215
215
|
async function runWithTimeout(timeoutInMilliseconds, fn, context, stackTrace) {
|
|
216
|
-
class Wrapper {
|
|
217
|
-
/**
|
|
218
|
-
* Creates a new wrapper.
|
|
219
|
-
*
|
|
220
|
-
* @param value - The value to wrap.
|
|
221
|
-
*/
|
|
222
|
-
constructor(value) {
|
|
223
|
-
this.value = value;
|
|
224
|
-
}
|
|
225
|
-
}
|
|
226
216
|
stackTrace ??= getStackTrace(1);
|
|
227
217
|
const startTime = performance.now();
|
|
228
|
-
const
|
|
218
|
+
const runAbortController = new AbortController();
|
|
219
|
+
const timeoutAbortController = new AbortController();
|
|
220
|
+
let result = null;
|
|
221
|
+
let hasResult = false;
|
|
229
222
|
await Promise.race([run(), innerTimeout()]);
|
|
230
|
-
if (
|
|
231
|
-
return
|
|
223
|
+
if (hasResult) {
|
|
224
|
+
return result;
|
|
232
225
|
}
|
|
233
|
-
throw new CustomStackTraceError("Run with timeout failed", stackTrace,
|
|
226
|
+
throw new CustomStackTraceError("Run with timeout failed", stackTrace, runAbortController.signal.reason);
|
|
234
227
|
async function run() {
|
|
235
228
|
try {
|
|
236
|
-
|
|
229
|
+
result = await fn(runAbortController.signal);
|
|
237
230
|
const duration = performance.now() - startTime;
|
|
238
231
|
const runWithTimeoutDebugger = getLibDebugger("Async:runWithTimeout");
|
|
239
232
|
printWithStackTrace(runWithTimeoutDebugger, stackTrace ?? "", `Execution time: ${String(duration)} milliseconds`, { context, fn });
|
|
240
|
-
|
|
233
|
+
timeoutAbortController.abort(new Error("Run with timeout completed successfully"));
|
|
234
|
+
hasResult = true;
|
|
241
235
|
} catch (e) {
|
|
242
|
-
|
|
236
|
+
runAbortController.abort(e);
|
|
243
237
|
}
|
|
244
238
|
}
|
|
245
239
|
async function innerTimeout() {
|
|
246
|
-
while (!
|
|
247
|
-
await sleep(timeoutInMilliseconds,
|
|
248
|
-
if (
|
|
240
|
+
while (!hasResult) {
|
|
241
|
+
await sleep(timeoutInMilliseconds, timeoutAbortController.signal);
|
|
242
|
+
if (hasResult) {
|
|
249
243
|
return;
|
|
250
244
|
}
|
|
251
245
|
const duration = performance.now() - startTime;
|
|
252
246
|
console.warn(`Timed out after ${String(duration)} milliseconds`, { context, fn });
|
|
253
247
|
const timeoutDebugger = getLibDebugger("Async:runWithTimeout:timeout");
|
|
254
248
|
if (!timeoutDebugger.enabled) {
|
|
255
|
-
|
|
249
|
+
runAbortController.abort(new Error(`Timed out after ${String(duration)} milliseconds`));
|
|
256
250
|
return;
|
|
257
251
|
}
|
|
258
252
|
timeoutDebugger(
|
|
@@ -316,4 +310,4 @@ export {
|
|
|
316
310
|
timeout,
|
|
317
311
|
toArray
|
|
318
312
|
};
|
|
319
|
-
//# 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';\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\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 * 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\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\nfunction 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\nconst terminateRetryErrors = new WeakSet<Error>();\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 fn - The function to retry.\n * @param retryOptions - Optional parameters to configure the retry behavior.\n * @param stackTrace - Optional stack trace.\n * @returns A {@link Promise} that resolves when the function returns true or rejects when the timeout is reached.\n */\nexport async function retryWithTimeout(fn: (abortSignal: AbortSignal) => Promisable<boolean>, retryOptions?: RetryOptions, stackTrace?: string): Promise<void> {\n  const retryWithTimeoutDebugger = getLibDebugger('Async:retryWithTimeout');\n  stackTrace ??= getStackTrace(1);\n  const DEFAULT_RETRY_OPTIONS = {\n    // eslint-disable-next-line no-magic-numbers\n    retryDelayInMilliseconds: 100,\n    shouldRetryOnError: false,\n    // eslint-disable-next-line no-magic-numbers\n    timeoutInMilliseconds: 5000\n  };\n  const fullOptions = { ...DEFAULT_RETRY_OPTIONS, ...retryOptions };\n  fullOptions.abortSignal?.throwIfAborted();\n\n  await runWithTimeout(\n    fullOptions.timeoutInMilliseconds,\n    async (abortSignal: AbortSignal) => {\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 fn(combinedAbortSignal);\n        } catch (error) {\n          // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\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            fn\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            fn\n          }\n        );\n\n        await sleep(fullOptions.retryDelayInMilliseconds, abortSignal);\n      }\n    },\n    { retryFn: fn },\n    stackTrace\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 R - The type of the result from the asynchronous function.\n * @param timeoutInMilliseconds - The maximum time to wait in milliseconds.\n * @param fn - The function to execute.\n * @param context - The context of the function.\n * @param stackTrace - The stack trace of the source 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>(\n  timeoutInMilliseconds: number,\n  fn: (abortSignal: AbortSignal) => Promisable<Result>,\n  context?: unknown,\n  stackTrace?: string\n): Promise<Result> {\n  class Wrapper {\n    /**\n     * Creates a new wrapper.\n     *\n     * @param value - The value to wrap.\n     */\n    public constructor(public readonly value: Result) {}\n  }\n\n  stackTrace ??= getStackTrace(1);\n  const startTime = performance.now();\n\n  const abortController = new AbortController();\n\n  await Promise.race([run(), innerTimeout()]);\n  if (abortController.signal.reason instanceof Wrapper) {\n    return (abortController.signal.reason as Wrapper).value;\n  }\n\n  throw new CustomStackTraceError('Run with timeout failed', stackTrace, abortController.signal.reason as unknown);\n\n  async function run(): Promise<void> {\n    try {\n      const result = await fn(abortController.signal);\n      const duration = performance.now() - startTime;\n      const runWithTimeoutDebugger = getLibDebugger('Async:runWithTimeout');\n      printWithStackTrace(runWithTimeoutDebugger, stackTrace ?? '', `Execution time: ${String(duration)} milliseconds`, { context, fn });\n      abortController.abort(new Wrapper(result));\n    } catch (e) {\n      abortController.abort(e);\n    }\n  }\n\n  async function innerTimeout(): Promise<void> {\n    while (!abortController.signal.aborted) {\n      await sleep(timeoutInMilliseconds, abortController.signal);\n      // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n      if (abortController.signal.aborted) {\n        return;\n      }\n      const duration = performance.now() - startTime;\n      console.warn(`Timed out after ${String(duration)} milliseconds`, { context, fn });\n      const timeoutDebugger = getLibDebugger('Async:runWithTimeout:timeout');\n      if (!timeoutDebugger.enabled) {\n        abortController.abort(new Error(`Timed out after ${String(duration)} milliseconds`));\n        return;\n      }\n\n      timeoutDebugger(\n        `The execution is not terminated because debugger ${timeoutDebugger.namespace} is enabled. See https://github.com/mnaoumov/obsidian-dev-utils/blob/main/docs/debugging.md for more information.`\n      );\n    }\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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",
  "names": []
}

|
|
313
|
+
//# 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';\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\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 * 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\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\nfunction 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\nconst terminateRetryErrors = new WeakSet<Error>();\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 fn - The function to retry.\n * @param retryOptions - Optional parameters to configure the retry behavior.\n * @param stackTrace - Optional stack trace.\n * @returns A {@link Promise} that resolves when the function returns true or rejects when the timeout is reached.\n */\nexport async function retryWithTimeout(fn: (abortSignal: AbortSignal) => Promisable<boolean>, retryOptions?: RetryOptions, stackTrace?: string): Promise<void> {\n  const retryWithTimeoutDebugger = getLibDebugger('Async:retryWithTimeout');\n  stackTrace ??= getStackTrace(1);\n  const DEFAULT_RETRY_OPTIONS = {\n    // eslint-disable-next-line no-magic-numbers\n    retryDelayInMilliseconds: 100,\n    shouldRetryOnError: false,\n    // eslint-disable-next-line no-magic-numbers\n    timeoutInMilliseconds: 5000\n  };\n  const fullOptions = { ...DEFAULT_RETRY_OPTIONS, ...retryOptions };\n  fullOptions.abortSignal?.throwIfAborted();\n\n  await runWithTimeout(\n    fullOptions.timeoutInMilliseconds,\n    async (abortSignal: AbortSignal) => {\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 fn(combinedAbortSignal);\n        } catch (error) {\n          // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\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            fn\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            fn\n          }\n        );\n\n        await sleep(fullOptions.retryDelayInMilliseconds, abortSignal);\n      }\n    },\n    { retryFn: fn },\n    stackTrace\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 R - The type of the result from the asynchronous function.\n * @param timeoutInMilliseconds - The maximum time to wait in milliseconds.\n * @param fn - The function to execute.\n * @param context - The context of the function.\n * @param stackTrace - The stack trace of the source 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>(\n  timeoutInMilliseconds: number,\n  fn: (abortSignal: AbortSignal) => Promisable<Result>,\n  context?: unknown,\n  stackTrace?: string\n): Promise<Result> {\n  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\n  await Promise.race([run(), innerTimeout()]);\n  // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\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 fn(runAbortController.signal);\n      const duration = performance.now() - startTime;\n      const runWithTimeoutDebugger = getLibDebugger('Async:runWithTimeout');\n      printWithStackTrace(runWithTimeoutDebugger, stackTrace ?? '', `Execution time: ${String(duration)} milliseconds`, { context, fn });\n      timeoutAbortController.abort(new Error('Run with timeout completed successfully'));\n      hasResult = true;\n    } catch (e) {\n      runAbortController.abort(e);\n    }\n  }\n\n  async function innerTimeout(): Promise<void> {\n    // eslint-disable-next-line no-unmodified-loop-condition\n    while (!hasResult) {\n      await sleep(timeoutInMilliseconds, timeoutAbortController.signal);\n      // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n      if (hasResult) {\n        return;\n      }\n      const duration = performance.now() - startTime;\n      console.warn(`Timed out after ${String(duration)} milliseconds`, { context, fn });\n      const timeoutDebugger = getLibDebugger('Async:runWithTimeout:timeout');\n      if (!timeoutDebugger.enabled) {\n        runAbortController.abort(new Error(`Timed out after ${String(duration)} milliseconds`));\n        return;\n      }\n\n      timeoutDebugger(\n        `The execution is not terminated because debugger ${timeoutDebugger.namespace} is enabled. See https://github.com/mnaoumov/obsidian-dev-utils/blob/main/docs/debugging.md for more information.`\n      );\n    }\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": ";;;;;;;AAQA;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP;AAAA,EACE;AAAA,EACA;AAAA,OACK;AACP;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP,SAAS,YAAY;AA0CrB,eAAsB,gBAAgB,SAAiC,YAAoC;AACzG,iBAAe,cAAc,CAAC;AAC9B,MAAI;AACF,UAAM,QAAQ;AAAA,EAChB,SAAS,YAAY;AACnB,UAAM,eAAe,IAAI,sBAAsB,6BAA6B,YAAY,UAAU;AAClG,QAAI,kBAAkB,YAAY,GAAG;AACnC;AAAA,IACF;AACA,wBAAoB,YAAY;AAAA,EAClC;AACF;AAUA,eAAsB,YAAe,KAAU,WAAuF;AACpI,QAAM,MAAW,CAAC;AAElB,QAAM,SAAS,IAAI;AACnB,WAAS,IAAI,GAAG,IAAI,QAAQ,KAAK;AAC/B,QAAI,CAAC,OAAO,OAAO,KAAK,CAAC,GAAG;AAC1B;AAAA,IACF;AAEA,UAAM,OAAO,IAAI,CAAC;AAClB,QAAI,MAAM,UAAU,MAAM,GAAG,GAAG,GAAG;AACjC,UAAI,KAAK,IAAI;AAAA,IACf;AAAA,EACF;AAEA,SAAO;AACT;AAUA,eAAsB,mBAAsB,KAAU,WAAwF;AAC5I,QAAM,SAAS,IAAI;AACnB,MAAI,aAAa;AACjB,WAAS,YAAY,GAAG,YAAY,QAAQ,aAAa;AACvD,QAAI,CAAC,OAAO,OAAO,KAAK,SAAS,GAAG;AAClC;AAAA,IACF;AAEA,UAAM,UAAU,IAAI,SAAS;AAC7B,QAAI,MAAM,UAAU,SAAS,WAAW,GAAG,GAAG;AAE5C,UAAI,YAAY,IAAI;AAAA,IACtB;AAAA,EACF;AACA,MAAI,SAAS;AACf;AAWA,eAAsB,aAAmB,KAAU,UAAkF;AACnI,UAAQ,MAAM,SAAS,KAAK,QAAQ,GAAG,KAAK;AAC9C;AAWA,eAAsB,SAAe,KAAU,UAAgF;AAC7H,SAAO,MAAM,uBAAuB,IAAI,IAAI,QAAQ,CAAC;AACvD;AAUO,SAAS,mBAA2C,WAAgD,YAA8C;AACvJ,iBAAe,cAAc,CAAC;AAC9B,SAAO,IAAI,SAAqB;AAC9B,UAAM,kBAAkB,cAAc,CAAC;AACvC,iBAAa,GAAG,cAAc,EAAE;AAAA;AAAA,EAA4C,eAAe;AAC3F,sBAAkB,MAAM,UAAU,GAAG,IAAI,GAAG,UAAU;AAAA,EACxD;AACF;AAUO,SAAS,mBAAmD,QAAuE;AACxI,SAAO,UAAU,SAAgC;AAC/C,UAAM,QAAQ,QAAQ;AACtB,WAAO,OAAO,GAAG,IAAI;AAAA,EACvB;AACF;AAmBA,eAAsB,YAAe,SAAqB,eAA8B;AACtF,QAAM,sBAAsB,eAAe,mBAAmB;AAC9D,QAAM,aAAa,cAAc,CAAC;AAClC,MAAI;AACF,WAAO,MAAM;AAAA,EACf,SAAS,GAAG;AACV,wBAAoB,iBAAiB,IAAI,sBAAsB,iBAAiB,YAAY,CAAC,CAAC;AAC9F,WAAO;AAAA,EACT;AACF;AAQO,SAAS,kBAAkB,SAAiC,YAA2B;AAC5F,iBAAe,cAAc,CAAC;AAE9B,OAAK,gBAAgB,SAAS,UAAU;AAC1C;AAUO,SAAS,4BACd,SACA,sBAAsB,GACtB,YACA,aACM;AACN,kBAAgB,iBAAiB;AACjC,cAAY,eAAe;AAC3B,iBAAe,cAAc,CAAC;AAC9B,oBAAkB,YAAY;AAC5B,UAAM,MAAM,qBAAqB,aAAa,IAAI;AAClD,UAAM,QAAQ,WAAW;AAAA,EAC3B,GAAG,UAAU;AACf;AASA,eAAsB,+BAAkC,UAAiD;AACvG,QAAM,UAAe,CAAC;AACtB,aAAW,WAAW,UAAU;AAC9B,YAAQ,KAAK,MAAM,QAAQ,CAAC;AAAA,EAC9B;AACA,SAAO;AACT;AASA,eAAsB,uBAA0B,UAAyC;AACvF,SAAO,MAAM,+BAA+B,SAAS,IAAI,CAAC,YAAY,MAAM,OAAO,CAAC;AACtF;AAEA,SAAS,kBAAkB,OAAyB;AAClD,MAAI,QAAQ;AACZ,SAAO,EAAE,iBAAiB,cAAc;AACtC,QAAI,EAAE,iBAAiB,QAAQ;AAC7B,aAAO;AAAA,IACT;AAEA,YAAQ,MAAM;AAAA,EAChB;AAEA,iBAAe,yBAAyB,EAAE,KAAK;AAC/C,SAAO;AACT;AAEA,MAAM,uBAAuB,oBAAI,QAAe;AAOzC,SAAS,sBAAsB,OAAoB;AACxD,uBAAqB,IAAI,KAAK;AAChC;AAOA,eAAsB,YAA4B;AAChD,QAAM,IAAI,QAAQ,MAAM;AACtB,SAAK;AAAA,EACP,CAAC;AACD,QAAM,IAAI,MAAM,qBAAqB;AACvC;AAOA,eAAsB,gBAA+B;AACnD,SAAO,IAAI,QAAQ,CAAC,YAAY;AAC9B,YAAQ,SAAS,MAAM;AACrB,cAAQ;AAAA,IACV,CAAC;AAAA,EACH,CAAC;AACH;AAOA,eAAsB,sBAAqC;AACzD,SAAO,IAAI,QAAQ,CAAC,YAAY;AAC9B,mBAAe,MAAM;AACnB,cAAQ;AAAA,IACV,CAAC;AAAA,EACH,CAAC;AACH;AAOA,eAAsB,6BAA4C;AAChE,SAAO,IAAI,QAAQ,CAAC,YAAY;AAC9B,0BAAsB,MAAM;AAC1B,cAAQ;AAAA,IACV,CAAC;AAAA,EACH,CAAC;AACH;AAUA,eAAsB,iBAAiB,IAAuD,cAA6B,YAAoC;AAC7J,QAAM,2BAA2B,eAAe,wBAAwB;AACxE,iBAAe,cAAc,CAAC;AAC9B,QAAM,wBAAwB;AAAA;AAAA,IAE5B,0BAA0B;AAAA,IAC1B,oBAAoB;AAAA;AAAA,IAEpB,uBAAuB;AAAA,EACzB;AACA,QAAM,cAAc,EAAE,GAAG,uBAAuB,GAAG,aAAa;AAChE,cAAY,aAAa,eAAe;AAExC,QAAM;AAAA,IACJ,YAAY;AAAA,IACZ,OAAO,gBAA6B;AAClC,YAAM,sBAAsB,eAAe,YAAY,aAAa,WAAW;AAC/E,0BAAoB,eAAe;AACnC,UAAI,UAAU;AACd,aAAO,CAAC,oBAAoB,SAAS;AACnC;AACA,YAAI;AACJ,YAAI;AACF,sBAAY,MAAM,GAAG,mBAAmB;AAAA,QAC1C,SAAS,OAAO;AAEd,cAAI,oBAAoB,WAAW,CAAC,YAAY,sBAAsB,qBAAqB,IAAI,KAAc,GAAG;AAC9G,kBAAM,IAAI,sBAAsB,2BAA2B,YAAY,KAAK;AAAA,UAC9E;AACA,qBAAW,KAAK;AAChB,sBAAY;AAAA,QACd;AACA,YAAI,WAAW;AACb,8BAAoB,0BAA0B,YAAY,sCAAsC,OAAO,OAAO,CAAC,aAAa;AAAA,YAC1H;AAAA,UACF,CAAC;AACD;AAAA,QACF;AAEA;AAAA,UACE;AAAA,UACA;AAAA,UACA,iBAAiB,OAAO,OAAO,CAAC,8CAA8C,OAAO,YAAY,wBAAwB,CAAC;AAAA,UAC1H;AAAA,YACE;AAAA,UACF;AAAA,QACF;AAEA,cAAM,MAAM,YAAY,0BAA0B,WAAW;AAAA,MAC/D;AAAA,IACF;AAAA,IACA,EAAE,SAAS,GAAG;AAAA,IACd;AAAA,EACF;AACF;AAcA,eAAsB,eACpB,uBACA,IACA,SACA,YACiB;AACjB,iBAAe,cAAc,CAAC;AAC9B,QAAM,YAAY,YAAY,IAAI;AAElC,QAAM,qBAAqB,IAAI,gBAAgB;AAC/C,QAAM,yBAAyB,IAAI,gBAAgB;AAEnD,MAAI,SAAwB;AAC5B,MAAI,YAAY;AAEhB,QAAM,QAAQ,KAAK,CAAC,IAAI,GAAG,aAAa,CAAC,CAAC;AAE1C,MAAI,WAAW;AACb,WAAO;AAAA,EACT;AAEA,QAAM,IAAI,sBAAsB,2BAA2B,YAAY,mBAAmB,OAAO,MAAM;AAEvG,iBAAe,MAAqB;AAClC,QAAI;AACF,eAAS,MAAM,GAAG,mBAAmB,MAAM;AAC3C,YAAM,WAAW,YAAY,IAAI,IAAI;AACrC,YAAM,yBAAyB,eAAe,sBAAsB;AACpE,0BAAoB,wBAAwB,cAAc,IAAI,mBAAmB,OAAO,QAAQ,CAAC,iBAAiB,EAAE,SAAS,GAAG,CAAC;AACjI,6BAAuB,MAAM,IAAI,MAAM,yCAAyC,CAAC;AACjF,kBAAY;AAAA,IACd,SAAS,GAAG;AACV,yBAAmB,MAAM,CAAC;AAAA,IAC5B;AAAA,EACF;AAEA,iBAAe,eAA8B;AAE3C,WAAO,CAAC,WAAW;AACjB,YAAM,MAAM,uBAAuB,uBAAuB,MAAM;AAEhE,UAAI,WAAW;AACb;AAAA,MACF;AACA,YAAM,WAAW,YAAY,IAAI,IAAI;AACrC,cAAQ,KAAK,mBAAmB,OAAO,QAAQ,CAAC,iBAAiB,EAAE,SAAS,GAAG,CAAC;AAChF,YAAM,kBAAkB,eAAe,8BAA8B;AACrE,UAAI,CAAC,gBAAgB,SAAS;AAC5B,2BAAmB,MAAM,IAAI,MAAM,mBAAmB,OAAO,QAAQ,CAAC,eAAe,CAAC;AACtF;AAAA,MACF;AAEA;AAAA,QACE,oDAAoD,gBAAgB,SAAS;AAAA,MAC/E;AAAA,IACF;AAAA,EACF;AACF;AAOA,eAAsB,oBAAmC;AACvD,SAAO,IAAI,QAAQ,CAAC,YAAY;AAC9B,iBAAa,MAAM;AACjB,cAAQ;AAAA,IACV,CAAC;AAAA,EACH,CAAC;AACH;AAQA,eAAsB,gBAAgB,OAA+B;AACnE,QAAM,IAAI,QAAQ,CAAC,YAAY;AAC7B,eAAW,SAAS,KAAK;AAAA,EAC3B,CAAC;AACH;AAUA,eAAsB,MAAM,cAAsB,aAA2B,oBAA6C;AACxH,QAAM,aAAa,eAAe,aAAa,mBAAmB,YAAY,CAAC,CAAC;AAChF,MAAI,oBAAoB;AACtB,iBAAa,eAAe;AAAA,EAC9B;AACF;AAUA,eAAsB,QAAQ,uBAA+B,aAA2B,oBAA8C;AACpI,QAAM,MAAM,uBAAuB,aAAa,kBAAkB;AAClE,QAAM,IAAI,MAAM,gBAAgB,OAAO,qBAAqB,CAAC,eAAe;AAC9E;AASA,eAAsB,QAAW,MAA8C;AAC7E,QAAM,MAAW,CAAC;AAClB,mBAAiB,QAAQ,MAAM;AAC7B,QAAI,KAAK,IAAI;AAAA,EACf;AACA,SAAO;AACT;",
  "names": []
}

|