obsidian-dev-utils 44.3.0 → 45.0.0
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 +4 -0
- package/dist/lib/cjs/Async.cjs +53 -32
- package/dist/lib/cjs/Async.d.cts +92 -10
- package/dist/lib/cjs/Library.cjs +1 -1
- package/dist/lib/cjs/obsidian/AsyncWithNotice.cjs +167 -0
- package/dist/lib/cjs/obsidian/AsyncWithNotice.d.cts +74 -0
- package/dist/lib/cjs/obsidian/Callout.cjs +8 -2
- package/dist/lib/cjs/obsidian/MetadataCache.cjs +45 -41
- package/dist/lib/cjs/obsidian/Queue.cjs +34 -17
- package/dist/lib/cjs/obsidian/Queue.d.cts +62 -12
- package/dist/lib/cjs/obsidian/RenameDeleteHandler.cjs +37 -27
- package/dist/lib/cjs/obsidian/Vault.cjs +45 -66
- package/dist/lib/cjs/obsidian/Vault.d.cts +0 -8
- package/dist/lib/cjs/obsidian/index.cjs +4 -1
- package/dist/lib/cjs/obsidian/index.d.cts +1 -0
- package/dist/lib/esm/Async.d.mts +92 -10
- package/dist/lib/esm/Async.mjs +53 -32
- package/dist/lib/esm/Library.mjs +1 -1
- package/dist/lib/esm/obsidian/AsyncWithNotice.d.mts +74 -0
- package/dist/lib/esm/obsidian/AsyncWithNotice.mjs +63 -0
- package/dist/lib/esm/obsidian/Callout.mjs +8 -2
- package/dist/lib/esm/obsidian/MetadataCache.mjs +46 -41
- package/dist/lib/esm/obsidian/Queue.d.mts +62 -12
- package/dist/lib/esm/obsidian/Queue.mjs +35 -19
- package/dist/lib/esm/obsidian/RenameDeleteHandler.mjs +37 -27
- package/dist/lib/esm/obsidian/Vault.d.mts +0 -8
- package/dist/lib/esm/obsidian/Vault.mjs +46 -70
- package/dist/lib/esm/obsidian/index.d.mts +1 -0
- package/dist/lib/esm/obsidian/index.mjs +3 -1
- package/obsidian/AsyncWithNotice/package.json +6 -0
- package/package.json +7 -7
|
@@ -25,13 +25,14 @@ import {
|
|
|
25
25
|
isReferenceCache,
|
|
26
26
|
parentFolderPath
|
|
27
27
|
} from "obsidian-typings/implementations";
|
|
28
|
-
import { retryWithTimeout } from "../Async.mjs";
|
|
29
28
|
import { getNestedPropertyValue } from "../ObjectUtils.mjs";
|
|
30
29
|
import { getObsidianDevUtilsState } from "./App.mjs";
|
|
30
|
+
import { retryWithTimeoutNotice } from "./AsyncWithNotice.mjs";
|
|
31
31
|
import {
|
|
32
32
|
getFile,
|
|
33
33
|
getFileOrNull,
|
|
34
34
|
getFolder,
|
|
35
|
+
getPath,
|
|
35
36
|
isFile
|
|
36
37
|
} from "./FileSystem.mjs";
|
|
37
38
|
import { parseFrontmatter } from "./Frontmatter.mjs";
|
|
@@ -91,51 +92,55 @@ async function getBacklinksForFileSafe(app, pathOrFile, retryOptions = {}) {
|
|
|
91
92
|
return safeOverload(pathOrFile);
|
|
92
93
|
}
|
|
93
94
|
let backlinks = new CustomArrayDictImpl();
|
|
94
|
-
await
|
|
95
|
-
abortSignal
|
|
96
|
-
const file = getFile(app, pathOrFile);
|
|
97
|
-
await ensureMetadataCacheReady(app);
|
|
98
|
-
abortSignal.throwIfAborted();
|
|
99
|
-
backlinks = getBacklinksForFileOrPath(app, file);
|
|
100
|
-
for (const notePath of backlinks.keys()) {
|
|
95
|
+
await retryWithTimeoutNotice({
|
|
96
|
+
async operationFn(abortSignal) {
|
|
101
97
|
abortSignal.throwIfAborted();
|
|
102
|
-
const
|
|
103
|
-
|
|
104
|
-
return false;
|
|
105
|
-
}
|
|
106
|
-
await saveNote(app, note);
|
|
107
|
-
abortSignal.throwIfAborted();
|
|
108
|
-
const content = await readSafe(app, note);
|
|
98
|
+
const file = getFile(app, pathOrFile);
|
|
99
|
+
await ensureMetadataCacheReady(app);
|
|
109
100
|
abortSignal.throwIfAborted();
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
const propertyValue = getNestedPropertyValue(frontmatter, link.key);
|
|
124
|
-
if (typeof propertyValue !== "string") {
|
|
125
|
-
return false;
|
|
126
|
-
}
|
|
127
|
-
const linkWithOffsets = toFrontmatterLinkCacheWithOffsets(link);
|
|
128
|
-
actualLink = propertyValue.slice(linkWithOffsets.startOffset, linkWithOffsets.endOffset);
|
|
129
|
-
} else {
|
|
130
|
-
return true;
|
|
101
|
+
backlinks = getBacklinksForFileOrPath(app, file);
|
|
102
|
+
for (const notePath of backlinks.keys()) {
|
|
103
|
+
abortSignal.throwIfAborted();
|
|
104
|
+
const note = getFileOrNull(app, notePath);
|
|
105
|
+
if (!note) {
|
|
106
|
+
return false;
|
|
107
|
+
}
|
|
108
|
+
await saveNote(app, note);
|
|
109
|
+
abortSignal.throwIfAborted();
|
|
110
|
+
const content = await readSafe(app, note);
|
|
111
|
+
abortSignal.throwIfAborted();
|
|
112
|
+
if (!content) {
|
|
113
|
+
return false;
|
|
131
114
|
}
|
|
132
|
-
|
|
115
|
+
const frontmatter = parseFrontmatter(content);
|
|
116
|
+
const links = backlinks.get(notePath);
|
|
117
|
+
if (!links) {
|
|
133
118
|
return false;
|
|
134
119
|
}
|
|
120
|
+
for (const link of links) {
|
|
121
|
+
let actualLink;
|
|
122
|
+
if (isReferenceCache(link)) {
|
|
123
|
+
actualLink = content.slice(link.position.start.offset, link.position.end.offset);
|
|
124
|
+
} else if (isFrontmatterLinkCache(link)) {
|
|
125
|
+
const propertyValue = getNestedPropertyValue(frontmatter, link.key);
|
|
126
|
+
if (typeof propertyValue !== "string") {
|
|
127
|
+
return false;
|
|
128
|
+
}
|
|
129
|
+
const linkWithOffsets = toFrontmatterLinkCacheWithOffsets(link);
|
|
130
|
+
actualLink = propertyValue.slice(linkWithOffsets.startOffset, linkWithOffsets.endOffset);
|
|
131
|
+
} else {
|
|
132
|
+
return true;
|
|
133
|
+
}
|
|
134
|
+
if (actualLink !== link.original) {
|
|
135
|
+
return false;
|
|
136
|
+
}
|
|
137
|
+
}
|
|
135
138
|
}
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
+
return true;
|
|
140
|
+
},
|
|
141
|
+
operationName: `Get backlinks for ${getPath(app, pathOrFile)}`,
|
|
142
|
+
retryOptions
|
|
143
|
+
});
|
|
139
144
|
return backlinks;
|
|
140
145
|
}
|
|
141
146
|
async function getCacheSafe(app, fileOrPath) {
|
|
@@ -258,4 +263,4 @@ export {
|
|
|
258
263
|
unregisterFileCacheForNonExistingFile,
|
|
259
264
|
unregisterFiles
|
|
260
265
|
};
|
|
261
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/MetadataCache.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utility functions for working with the metadata cache in Obsidian.\n */\n\nimport type {\n  App,\n  CachedMetadata,\n  Reference,\n  TAbstractFile\n} from 'obsidian';\nimport type { CustomArrayDict } from 'obsidian-typings';\n\nimport {\n  CustomArrayDictImpl,\n  isFrontmatterLinkCache,\n  isReferenceCache,\n  parentFolderPath\n} from 'obsidian-typings/implementations';\n\nimport type { RetryOptions } from '../Async.ts';\nimport type { PathOrFile } from './FileSystem.ts';\nimport type { CombinedFrontmatter } from './Frontmatter.ts';\n\nimport { retryWithTimeout } from '../Async.ts';\nimport { getNestedPropertyValue } from '../ObjectUtils.ts';\nimport { getObsidianDevUtilsState } from './App.ts';\nimport {\n  getFile,\n  getFileOrNull,\n  getFolder,\n  isFile\n} from './FileSystem.ts';\nimport { parseFrontmatter } from './Frontmatter.ts';\nimport {\n  isFrontmatterLinkCacheWithOffsets,\n  toFrontmatterLinkCacheWithOffsets\n} from './FrontmatterLinkCacheWithOffsets.ts';\nimport { sortReferences } from './Reference.ts';\nimport {\n  readSafe,\n  saveNote\n} from './Vault.ts';\n\n/**\n * Wrapper for the getBacklinksForFile method that provides a safe overload.\n */\nexport interface GetBacklinksForFileSafeWrapper {\n  /**\n   * Retrieves the backlinks for a file safely.\n   *\n   * @param pathOrFile - The path or file object.\n   * @returns A {@link Promise} that resolves to an array dictionary of backlinks.\n   */\n  safe(pathOrFile: PathOrFile): Promise<CustomArrayDict<Reference>>;\n}\n\n/**\n * Ensures that the metadata cache is ready for all files.\n *\n * @param app - The Obsidian app instance.\n * @returns A {@link Promise} that resolves when the metadata cache is ready.\n */\nexport async function ensureMetadataCacheReady(app: App): Promise<void> {\n  await new Promise((resolve) => {\n    app.metadataCache.onCleanCache(resolve);\n  });\n}\n\n/**\n * Retrieves all links from the provided cache.\n *\n * @param cache - The cached metadata.\n * @returns An array of reference caches representing the links.\n */\nexport function getAllLinks(cache: CachedMetadata): Reference[] {\n  let links: Reference[] = [];\n\n  if (cache.links) {\n    links.push(...cache.links);\n  }\n\n  if (cache.embeds) {\n    links.push(...cache.embeds);\n  }\n\n  if (cache.frontmatterLinks) {\n    links.push(...cache.frontmatterLinks);\n  }\n\n  sortReferences(links);\n\n  // BUG: https://forum.obsidian.md/t/bug-duplicated-links-in-metadatacache-inside-footnotes/85551\n  links = links.filter((link, index) => {\n    if (index === 0) {\n      return true;\n    }\n\n    const previousLink = links[index - 1];\n    if (!previousLink) {\n      return true;\n    }\n\n    if (isReferenceCache(link) && isReferenceCache(previousLink)) {\n      return link.position.start.offset !== previousLink.position.start.offset;\n    }\n\n    if (isFrontmatterLinkCache(link) && isFrontmatterLinkCache(previousLink)) {\n      const linkStartOffset = isFrontmatterLinkCacheWithOffsets(link) ? link.startOffset : 0;\n      const previousLinkStartOffset = isFrontmatterLinkCacheWithOffsets(previousLink) ? previousLink.startOffset : 0;\n      return link.key !== previousLink.key || isFrontmatterLinkCacheWithOffsets(link) !== isFrontmatterLinkCacheWithOffsets(previousLink)\n        || linkStartOffset !== previousLinkStartOffset;\n    }\n\n    return true;\n  });\n\n  return links;\n}\n\n/**\n * Retrieves the backlinks for a file or path.\n * NOTE: The file may be non-existent.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or file object.\n * @returns The backlinks for the file.\n */\nexport function getBacklinksForFileOrPath(app: App, pathOrFile: PathOrFile): CustomArrayDict<Reference> {\n  const file = getFile(app, pathOrFile, true);\n  return tempRegisterFilesAndRun(app, [file], () => app.metadataCache.getBacklinksForFile(file));\n}\n\n/**\n * Retrieves the backlinks for a file safely.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or file object.\n * @param retryOptions - Optional retry options.\n * @returns A {@link Promise} that resolves to an array dictionary of backlinks.\n */\nexport async function getBacklinksForFileSafe(app: App, pathOrFile: PathOrFile, retryOptions: RetryOptions = {}): Promise<CustomArrayDict<Reference>> {\n  const safeOverload = (app.metadataCache.getBacklinksForFile as Partial<GetBacklinksForFileSafeWrapper>).safe;\n  if (safeOverload) {\n    return safeOverload(pathOrFile);\n  }\n  let backlinks: CustomArrayDict<Reference> = new CustomArrayDictImpl<Reference>();\n  await retryWithTimeout(async (abortSignal) => {\n    abortSignal.throwIfAborted();\n    const file = getFile(app, pathOrFile);\n    await ensureMetadataCacheReady(app);\n    abortSignal.throwIfAborted();\n    backlinks = getBacklinksForFileOrPath(app, file);\n    for (const notePath of backlinks.keys()) {\n      abortSignal.throwIfAborted();\n      const note = getFileOrNull(app, notePath);\n      if (!note) {\n        return false;\n      }\n\n      await saveNote(app, note);\n      abortSignal.throwIfAborted();\n\n      const content = await readSafe(app, note);\n      abortSignal.throwIfAborted();\n      if (!content) {\n        return false;\n      }\n      const frontmatter = parseFrontmatter(content);\n      const links = backlinks.get(notePath);\n      if (!links) {\n        return false;\n      }\n\n      for (const link of links) {\n        let actualLink: string;\n        if (isReferenceCache(link)) {\n          actualLink = content.slice(link.position.start.offset, link.position.end.offset);\n        } else if (isFrontmatterLinkCache(link)) {\n          const propertyValue = getNestedPropertyValue(frontmatter, link.key);\n          if (typeof propertyValue !== 'string') {\n            return false;\n          }\n\n          const linkWithOffsets = toFrontmatterLinkCacheWithOffsets(link);\n          actualLink = propertyValue.slice(linkWithOffsets.startOffset, linkWithOffsets.endOffset);\n        } else {\n          return true;\n        }\n        if (actualLink !== link.original) {\n          return false;\n        }\n      }\n    }\n\n    return true;\n  }, retryOptions);\n\n  return backlinks;\n}\n\n/**\n * Retrieves the cached metadata for a given file or path.\n *\n * @param app - The Obsidian app instance.\n * @param fileOrPath - The file or path to retrieve the metadata for.\n * @returns The cached metadata for the file, or null if it doesn't exist.\n */\nexport async function getCacheSafe(app: App, fileOrPath: PathOrFile): Promise<CachedMetadata | null> {\n  const file = getFileOrNull(app, fileOrPath);\n\n  try {\n    if (!file) {\n      return null;\n    }\n\n    if (file.deleted) {\n      return app.metadataCache.getFileCache(file);\n    }\n\n    await saveNote(app, file);\n\n    const fileCacheEntry = app.metadataCache.fileCache[file.path];\n    const isUpToDate = fileCacheEntry\n      && fileCacheEntry.mtime === file.stat.mtime\n      && fileCacheEntry.size === file.stat.size\n      && app.metadataCache.metadataCache[fileCacheEntry.hash];\n    if (!isUpToDate) {\n      await app.metadataCache.computeFileMetadataAsync(file);\n      await ensureMetadataCacheReady(app);\n    }\n    return app.metadataCache.getFileCache(file);\n  } catch (error) {\n    if (!file || file.deleted) {\n      return null;\n    }\n\n    throw error;\n  }\n}\n\n/**\n * Retrieves the front matter from the metadata cache safely.\n *\n * @typeParam CustomFrontmatter - The type of custom front matter.\n * @param app - The Obsidian app instance.\n * @param pathOrFile - The path or file to retrieve the front matter from.\n * @returns The combined front matter.\n */\nexport async function getFrontmatterSafe<CustomFrontmatter = unknown>(app: App, pathOrFile: PathOrFile): Promise<CombinedFrontmatter<CustomFrontmatter>> {\n  const cache = await getCacheSafe(app, pathOrFile);\n  return (cache?.frontmatter ?? {}) as CombinedFrontmatter<CustomFrontmatter>;\n}\n\n/**\n * Parses the metadata for a given string.\n *\n * @param app - The Obsidian app instance.\n * @param str - The string to parse the metadata for.\n * @returns The parsed metadata.\n */\nexport async function parseMetadata(app: App, str: string): Promise<CachedMetadata> {\n  const encoder = new TextEncoder();\n  const buffer = encoder.encode(str).buffer;\n  return await app.metadataCache.computeMetadataAsync(buffer) ?? {};\n}\n\n/**\n * Registers the file cache for a non-existing file.\n *\n * @param app - The Obsidian app instance.\n * @param pathOrFile - The path or file to register the file cache for.\n * @param cache - The file cache to register.\n */\nexport function registerFileCacheForNonExistingFile(app: App, pathOrFile: PathOrFile, cache: CachedMetadata): void {\n  const file = getFile(app, pathOrFile, true);\n  if (!file.deleted) {\n    throw new Error('File is existing');\n  }\n\n  app.metadataCache.fileCache[file.path] = {\n    hash: file.path,\n    mtime: 0,\n    size: 0\n  };\n\n  app.metadataCache.metadataCache[file.path] = cache;\n}\n\n/**\n * Registers files in the Obsidian app.\n *\n * @param app - The Obsidian app instance.\n * @param files - The files to register.\n */\nexport function registerFiles(app: App, files: TAbstractFile[]): void {\n  const registeredFilesCounts = getRegisteredFilesCounts(app);\n\n  for (let file of files) {\n    while (file.deleted) {\n      let count = registeredFilesCounts.get(file.path) ?? 0;\n      count++;\n      registeredFilesCounts.set(file.path, count);\n\n      app.vault.fileMap[file.path] = file;\n\n      if (isFile(file)) {\n        app.metadataCache.uniqueFileLookup.add(file.name.toLowerCase(), file);\n      }\n\n      file = getFolder(app, parentFolderPath(file.path), true);\n    }\n  }\n}\n\n/**\n * Temporarily registers files and runs a function.\n *\n * @typeParam T - The type of the result of the function.\n * @param app - The Obsidian app instance.\n * @param files - The files to temporarily register.\n * @param fn - The function to run.\n * @returns The result of the function.\n */\nexport function tempRegisterFilesAndRun<T>(app: App, files: TAbstractFile[], fn: () => T): T {\n  try {\n    registerFiles(app, files);\n    return fn();\n  } finally {\n    unregisterFiles(app, files);\n  }\n}\n\n/**\n * Temporarily registers files and runs an async function.\n *\n * @typeParam T - The type of the result of the function.\n * @param app - The Obsidian app instance.\n * @param files - The files to temporarily register.\n * @param fn - The function to run.\n * @returns The result of the function.\n */\nexport async function tempRegisterFilesAndRunAsync<T>(app: App, files: TAbstractFile[], fn: () => Promise<T>): Promise<T> {\n  try {\n    registerFiles(app, files);\n    return await fn();\n  } finally {\n    unregisterFiles(app, files);\n  }\n}\n\n/**\n * Unregisters the file cache for a non-existing file.\n *\n * @param app - The Obsidian app instance.\n * @param pathOrFile - The path or file to unregister the file cache for.\n */\nexport function unregisterFileCacheForNonExistingFile(app: App, pathOrFile: PathOrFile): void {\n  const file = getFile(app, pathOrFile, true);\n  if (!file.deleted) {\n    throw new Error('File is existing');\n  }\n  // eslint-disable-next-line @typescript-eslint/no-dynamic-delete -- We have no other way to delete the property.\n  delete app.metadataCache.fileCache[file.path];\n  // eslint-disable-next-line @typescript-eslint/no-dynamic-delete -- We have no other way to delete the property.\n  delete app.metadataCache.metadataCache[file.path];\n}\n\n/**\n * Unregisters files from the Obsidian app.\n *\n * @param app - The Obsidian app instance.\n * @param files - The files to unregister.\n */\nexport function unregisterFiles(app: App, files: TAbstractFile[]): void {\n  const registeredFilesCounts = getRegisteredFilesCounts(app);\n\n  for (let file of files) {\n    while (file.deleted) {\n      let count = registeredFilesCounts.get(file.path) ?? 1;\n      count--;\n      registeredFilesCounts.set(file.path, count);\n      if (count === 0) {\n        registeredFilesCounts.delete(file.path);\n        // eslint-disable-next-line @typescript-eslint/no-dynamic-delete -- We have no other way to delete the property.\n        delete app.vault.fileMap[file.path];\n\n        if (isFile(file)) {\n          app.metadataCache.uniqueFileLookup.remove(file.name.toLowerCase(), file);\n        }\n      }\n\n      file = getFolder(app, parentFolderPath(file.path), true);\n    }\n  }\n}\n\nfunction getRegisteredFilesCounts(app: App): Map<string, number> {\n  return getObsidianDevUtilsState(app, 'registeredFilesCounts', new Map<string, number>()).value;\n}\n"],
  "mappings": "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",
  "names": []
}

|
|
266
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/MetadataCache.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utility functions for working with the metadata cache in Obsidian.\n */\n\nimport type {\n  App,\n  CachedMetadata,\n  Reference,\n  TAbstractFile\n} from 'obsidian';\nimport type { CustomArrayDict } from 'obsidian-typings';\n\nimport {\n  CustomArrayDictImpl,\n  isFrontmatterLinkCache,\n  isReferenceCache,\n  parentFolderPath\n} from 'obsidian-typings/implementations';\n\nimport type { RetryOptions } from '../Async.ts';\nimport type { PathOrFile } from './FileSystem.ts';\nimport type { CombinedFrontmatter } from './Frontmatter.ts';\n\nimport { getNestedPropertyValue } from '../ObjectUtils.ts';\nimport { getObsidianDevUtilsState } from './App.ts';\nimport { retryWithTimeoutNotice } from './AsyncWithNotice.ts';\nimport {\n  getFile,\n  getFileOrNull,\n  getFolder,\n  getPath,\n  isFile\n} from './FileSystem.ts';\nimport { parseFrontmatter } from './Frontmatter.ts';\nimport {\n  isFrontmatterLinkCacheWithOffsets,\n  toFrontmatterLinkCacheWithOffsets\n} from './FrontmatterLinkCacheWithOffsets.ts';\nimport { sortReferences } from './Reference.ts';\nimport {\n  readSafe,\n  saveNote\n} from './Vault.ts';\n\n/**\n * Wrapper for the getBacklinksForFile method that provides a safe overload.\n */\nexport interface GetBacklinksForFileSafeWrapper {\n  /**\n   * Retrieves the backlinks for a file safely.\n   *\n   * @param pathOrFile - The path or file object.\n   * @returns A {@link Promise} that resolves to an array dictionary of backlinks.\n   */\n  safe(pathOrFile: PathOrFile): Promise<CustomArrayDict<Reference>>;\n}\n\n/**\n * Ensures that the metadata cache is ready for all files.\n *\n * @param app - The Obsidian app instance.\n * @returns A {@link Promise} that resolves when the metadata cache is ready.\n */\nexport async function ensureMetadataCacheReady(app: App): Promise<void> {\n  await new Promise((resolve) => {\n    app.metadataCache.onCleanCache(resolve);\n  });\n}\n\n/**\n * Retrieves all links from the provided cache.\n *\n * @param cache - The cached metadata.\n * @returns An array of reference caches representing the links.\n */\nexport function getAllLinks(cache: CachedMetadata): Reference[] {\n  let links: Reference[] = [];\n\n  if (cache.links) {\n    links.push(...cache.links);\n  }\n\n  if (cache.embeds) {\n    links.push(...cache.embeds);\n  }\n\n  if (cache.frontmatterLinks) {\n    links.push(...cache.frontmatterLinks);\n  }\n\n  sortReferences(links);\n\n  // BUG: https://forum.obsidian.md/t/bug-duplicated-links-in-metadatacache-inside-footnotes/85551\n  links = links.filter((link, index) => {\n    if (index === 0) {\n      return true;\n    }\n\n    const previousLink = links[index - 1];\n    if (!previousLink) {\n      return true;\n    }\n\n    if (isReferenceCache(link) && isReferenceCache(previousLink)) {\n      return link.position.start.offset !== previousLink.position.start.offset;\n    }\n\n    if (isFrontmatterLinkCache(link) && isFrontmatterLinkCache(previousLink)) {\n      const linkStartOffset = isFrontmatterLinkCacheWithOffsets(link) ? link.startOffset : 0;\n      const previousLinkStartOffset = isFrontmatterLinkCacheWithOffsets(previousLink) ? previousLink.startOffset : 0;\n      return link.key !== previousLink.key || isFrontmatterLinkCacheWithOffsets(link) !== isFrontmatterLinkCacheWithOffsets(previousLink)\n        || linkStartOffset !== previousLinkStartOffset;\n    }\n\n    return true;\n  });\n\n  return links;\n}\n\n/**\n * Retrieves the backlinks for a file or path.\n * NOTE: The file may be non-existent.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or file object.\n * @returns The backlinks for the file.\n */\nexport function getBacklinksForFileOrPath(app: App, pathOrFile: PathOrFile): CustomArrayDict<Reference> {\n  const file = getFile(app, pathOrFile, true);\n  return tempRegisterFilesAndRun(app, [file], () => app.metadataCache.getBacklinksForFile(file));\n}\n\n/**\n * Retrieves the backlinks for a file safely.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or file object.\n * @param retryOptions - Optional retry options.\n * @returns A {@link Promise} that resolves to an array dictionary of backlinks.\n */\nexport async function getBacklinksForFileSafe(app: App, pathOrFile: PathOrFile, retryOptions: RetryOptions = {}): Promise<CustomArrayDict<Reference>> {\n  const safeOverload = (app.metadataCache.getBacklinksForFile as Partial<GetBacklinksForFileSafeWrapper>).safe;\n  if (safeOverload) {\n    return safeOverload(pathOrFile);\n  }\n  let backlinks: CustomArrayDict<Reference> = new CustomArrayDictImpl<Reference>();\n  await retryWithTimeoutNotice({\n    async operationFn(abortSignal) {\n      abortSignal.throwIfAborted();\n      const file = getFile(app, pathOrFile);\n      await ensureMetadataCacheReady(app);\n      abortSignal.throwIfAborted();\n      backlinks = getBacklinksForFileOrPath(app, file);\n      for (const notePath of backlinks.keys()) {\n        abortSignal.throwIfAborted();\n        const note = getFileOrNull(app, notePath);\n        if (!note) {\n          return false;\n        }\n\n        await saveNote(app, note);\n        abortSignal.throwIfAborted();\n\n        const content = await readSafe(app, note);\n        abortSignal.throwIfAborted();\n        if (!content) {\n          return false;\n        }\n        const frontmatter = parseFrontmatter(content);\n        const links = backlinks.get(notePath);\n        if (!links) {\n          return false;\n        }\n\n        for (const link of links) {\n          let actualLink: string;\n          if (isReferenceCache(link)) {\n            actualLink = content.slice(link.position.start.offset, link.position.end.offset);\n          } else if (isFrontmatterLinkCache(link)) {\n            const propertyValue = getNestedPropertyValue(frontmatter, link.key);\n            if (typeof propertyValue !== 'string') {\n              return false;\n            }\n\n            const linkWithOffsets = toFrontmatterLinkCacheWithOffsets(link);\n            actualLink = propertyValue.slice(linkWithOffsets.startOffset, linkWithOffsets.endOffset);\n          } else {\n            return true;\n          }\n          if (actualLink !== link.original) {\n            return false;\n          }\n        }\n      }\n\n      return true;\n    },\n    operationName: `Get backlinks for ${getPath(app, pathOrFile)}`,\n    retryOptions\n  });\n\n  return backlinks;\n}\n\n/**\n * Retrieves the cached metadata for a given file or path.\n *\n * @param app - The Obsidian app instance.\n * @param fileOrPath - The file or path to retrieve the metadata for.\n * @returns The cached metadata for the file, or null if it doesn't exist.\n */\nexport async function getCacheSafe(app: App, fileOrPath: PathOrFile): Promise<CachedMetadata | null> {\n  const file = getFileOrNull(app, fileOrPath);\n\n  try {\n    if (!file) {\n      return null;\n    }\n\n    if (file.deleted) {\n      return app.metadataCache.getFileCache(file);\n    }\n\n    await saveNote(app, file);\n\n    const fileCacheEntry = app.metadataCache.fileCache[file.path];\n    const isUpToDate = fileCacheEntry\n      && fileCacheEntry.mtime === file.stat.mtime\n      && fileCacheEntry.size === file.stat.size\n      && app.metadataCache.metadataCache[fileCacheEntry.hash];\n    if (!isUpToDate) {\n      await app.metadataCache.computeFileMetadataAsync(file);\n      await ensureMetadataCacheReady(app);\n    }\n    return app.metadataCache.getFileCache(file);\n  } catch (error) {\n    if (!file || file.deleted) {\n      return null;\n    }\n\n    throw error;\n  }\n}\n\n/**\n * Retrieves the front matter from the metadata cache safely.\n *\n * @typeParam CustomFrontmatter - The type of custom front matter.\n * @param app - The Obsidian app instance.\n * @param pathOrFile - The path or file to retrieve the front matter from.\n * @returns The combined front matter.\n */\nexport async function getFrontmatterSafe<CustomFrontmatter = unknown>(app: App, pathOrFile: PathOrFile): Promise<CombinedFrontmatter<CustomFrontmatter>> {\n  const cache = await getCacheSafe(app, pathOrFile);\n  return (cache?.frontmatter ?? {}) as CombinedFrontmatter<CustomFrontmatter>;\n}\n\n/**\n * Parses the metadata for a given string.\n *\n * @param app - The Obsidian app instance.\n * @param str - The string to parse the metadata for.\n * @returns The parsed metadata.\n */\nexport async function parseMetadata(app: App, str: string): Promise<CachedMetadata> {\n  const encoder = new TextEncoder();\n  const buffer = encoder.encode(str).buffer;\n  return await app.metadataCache.computeMetadataAsync(buffer) ?? {};\n}\n\n/**\n * Registers the file cache for a non-existing file.\n *\n * @param app - The Obsidian app instance.\n * @param pathOrFile - The path or file to register the file cache for.\n * @param cache - The file cache to register.\n */\nexport function registerFileCacheForNonExistingFile(app: App, pathOrFile: PathOrFile, cache: CachedMetadata): void {\n  const file = getFile(app, pathOrFile, true);\n  if (!file.deleted) {\n    throw new Error('File is existing');\n  }\n\n  app.metadataCache.fileCache[file.path] = {\n    hash: file.path,\n    mtime: 0,\n    size: 0\n  };\n\n  app.metadataCache.metadataCache[file.path] = cache;\n}\n\n/**\n * Registers files in the Obsidian app.\n *\n * @param app - The Obsidian app instance.\n * @param files - The files to register.\n */\nexport function registerFiles(app: App, files: TAbstractFile[]): void {\n  const registeredFilesCounts = getRegisteredFilesCounts(app);\n\n  for (let file of files) {\n    while (file.deleted) {\n      let count = registeredFilesCounts.get(file.path) ?? 0;\n      count++;\n      registeredFilesCounts.set(file.path, count);\n\n      app.vault.fileMap[file.path] = file;\n\n      if (isFile(file)) {\n        app.metadataCache.uniqueFileLookup.add(file.name.toLowerCase(), file);\n      }\n\n      file = getFolder(app, parentFolderPath(file.path), true);\n    }\n  }\n}\n\n/**\n * Temporarily registers files and runs a function.\n *\n * @typeParam T - The type of the result of the function.\n * @param app - The Obsidian app instance.\n * @param files - The files to temporarily register.\n * @param fn - The function to run.\n * @returns The result of the function.\n */\nexport function tempRegisterFilesAndRun<T>(app: App, files: TAbstractFile[], fn: () => T): T {\n  try {\n    registerFiles(app, files);\n    return fn();\n  } finally {\n    unregisterFiles(app, files);\n  }\n}\n\n/**\n * Temporarily registers files and runs an async function.\n *\n * @typeParam T - The type of the result of the function.\n * @param app - The Obsidian app instance.\n * @param files - The files to temporarily register.\n * @param fn - The function to run.\n * @returns The result of the function.\n */\nexport async function tempRegisterFilesAndRunAsync<T>(app: App, files: TAbstractFile[], fn: () => Promise<T>): Promise<T> {\n  try {\n    registerFiles(app, files);\n    return await fn();\n  } finally {\n    unregisterFiles(app, files);\n  }\n}\n\n/**\n * Unregisters the file cache for a non-existing file.\n *\n * @param app - The Obsidian app instance.\n * @param pathOrFile - The path or file to unregister the file cache for.\n */\nexport function unregisterFileCacheForNonExistingFile(app: App, pathOrFile: PathOrFile): void {\n  const file = getFile(app, pathOrFile, true);\n  if (!file.deleted) {\n    throw new Error('File is existing');\n  }\n  // eslint-disable-next-line @typescript-eslint/no-dynamic-delete -- We have no other way to delete the property.\n  delete app.metadataCache.fileCache[file.path];\n  // eslint-disable-next-line @typescript-eslint/no-dynamic-delete -- We have no other way to delete the property.\n  delete app.metadataCache.metadataCache[file.path];\n}\n\n/**\n * Unregisters files from the Obsidian app.\n *\n * @param app - The Obsidian app instance.\n * @param files - The files to unregister.\n */\nexport function unregisterFiles(app: App, files: TAbstractFile[]): void {\n  const registeredFilesCounts = getRegisteredFilesCounts(app);\n\n  for (let file of files) {\n    while (file.deleted) {\n      let count = registeredFilesCounts.get(file.path) ?? 1;\n      count--;\n      registeredFilesCounts.set(file.path, count);\n      if (count === 0) {\n        registeredFilesCounts.delete(file.path);\n        // eslint-disable-next-line @typescript-eslint/no-dynamic-delete -- We have no other way to delete the property.\n        delete app.vault.fileMap[file.path];\n\n        if (isFile(file)) {\n          app.metadataCache.uniqueFileLookup.remove(file.name.toLowerCase(), file);\n        }\n      }\n\n      file = getFolder(app, parentFolderPath(file.path), true);\n    }\n  }\n}\n\nfunction getRegisteredFilesCounts(app: App): Map<string, number> {\n  return getObsidianDevUtilsState(app, 'registeredFilesCounts', new Map<string, number>()).value;\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;AAcA;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AAMP,SAAS,8BAA8B;AACvC,SAAS,gCAAgC;AACzC,SAAS,8BAA8B;AACvC;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP,SAAS,wBAAwB;AACjC;AAAA,EACE;AAAA,EACA;AAAA,OACK;AACP,SAAS,sBAAsB;AAC/B;AAAA,EACE;AAAA,EACA;AAAA,OACK;AAqBP,eAAsB,yBAAyB,KAAyB;AACtE,QAAM,IAAI,QAAQ,CAAC,YAAY;AAC7B,QAAI,cAAc,aAAa,OAAO;AAAA,EACxC,CAAC;AACH;AAQO,SAAS,YAAY,OAAoC;AAC9D,MAAI,QAAqB,CAAC;AAE1B,MAAI,MAAM,OAAO;AACf,UAAM,KAAK,GAAG,MAAM,KAAK;AAAA,EAC3B;AAEA,MAAI,MAAM,QAAQ;AAChB,UAAM,KAAK,GAAG,MAAM,MAAM;AAAA,EAC5B;AAEA,MAAI,MAAM,kBAAkB;AAC1B,UAAM,KAAK,GAAG,MAAM,gBAAgB;AAAA,EACtC;AAEA,iBAAe,KAAK;AAGpB,UAAQ,MAAM,OAAO,CAAC,MAAM,UAAU;AACpC,QAAI,UAAU,GAAG;AACf,aAAO;AAAA,IACT;AAEA,UAAM,eAAe,MAAM,QAAQ,CAAC;AACpC,QAAI,CAAC,cAAc;AACjB,aAAO;AAAA,IACT;AAEA,QAAI,iBAAiB,IAAI,KAAK,iBAAiB,YAAY,GAAG;AAC5D,aAAO,KAAK,SAAS,MAAM,WAAW,aAAa,SAAS,MAAM;AAAA,IACpE;AAEA,QAAI,uBAAuB,IAAI,KAAK,uBAAuB,YAAY,GAAG;AACxE,YAAM,kBAAkB,kCAAkC,IAAI,IAAI,KAAK,cAAc;AACrF,YAAM,0BAA0B,kCAAkC,YAAY,IAAI,aAAa,cAAc;AAC7G,aAAO,KAAK,QAAQ,aAAa,OAAO,kCAAkC,IAAI,MAAM,kCAAkC,YAAY,KAC7H,oBAAoB;AAAA,IAC3B;AAEA,WAAO;AAAA,EACT,CAAC;AAED,SAAO;AACT;AAUO,SAAS,0BAA0B,KAAU,YAAoD;AACtG,QAAM,OAAO,QAAQ,KAAK,YAAY,IAAI;AAC1C,SAAO,wBAAwB,KAAK,CAAC,IAAI,GAAG,MAAM,IAAI,cAAc,oBAAoB,IAAI,CAAC;AAC/F;AAUA,eAAsB,wBAAwB,KAAU,YAAwB,eAA6B,CAAC,GAAwC;AACpJ,QAAM,eAAgB,IAAI,cAAc,oBAAgE;AACxG,MAAI,cAAc;AAChB,WAAO,aAAa,UAAU;AAAA,EAChC;AACA,MAAI,YAAwC,IAAI,oBAA+B;AAC/E,QAAM,uBAAuB;AAAA,IAC3B,MAAM,YAAY,aAAa;AAC7B,kBAAY,eAAe;AAC3B,YAAM,OAAO,QAAQ,KAAK,UAAU;AACpC,YAAM,yBAAyB,GAAG;AAClC,kBAAY,eAAe;AAC3B,kBAAY,0BAA0B,KAAK,IAAI;AAC/C,iBAAW,YAAY,UAAU,KAAK,GAAG;AACvC,oBAAY,eAAe;AAC3B,cAAM,OAAO,cAAc,KAAK,QAAQ;AACxC,YAAI,CAAC,MAAM;AACT,iBAAO;AAAA,QACT;AAEA,cAAM,SAAS,KAAK,IAAI;AACxB,oBAAY,eAAe;AAE3B,cAAM,UAAU,MAAM,SAAS,KAAK,IAAI;AACxC,oBAAY,eAAe;AAC3B,YAAI,CAAC,SAAS;AACZ,iBAAO;AAAA,QACT;AACA,cAAM,cAAc,iBAAiB,OAAO;AAC5C,cAAM,QAAQ,UAAU,IAAI,QAAQ;AACpC,YAAI,CAAC,OAAO;AACV,iBAAO;AAAA,QACT;AAEA,mBAAW,QAAQ,OAAO;AACxB,cAAI;AACJ,cAAI,iBAAiB,IAAI,GAAG;AAC1B,yBAAa,QAAQ,MAAM,KAAK,SAAS,MAAM,QAAQ,KAAK,SAAS,IAAI,MAAM;AAAA,UACjF,WAAW,uBAAuB,IAAI,GAAG;AACvC,kBAAM,gBAAgB,uBAAuB,aAAa,KAAK,GAAG;AAClE,gBAAI,OAAO,kBAAkB,UAAU;AACrC,qBAAO;AAAA,YACT;AAEA,kBAAM,kBAAkB,kCAAkC,IAAI;AAC9D,yBAAa,cAAc,MAAM,gBAAgB,aAAa,gBAAgB,SAAS;AAAA,UACzF,OAAO;AACL,mBAAO;AAAA,UACT;AACA,cAAI,eAAe,KAAK,UAAU;AAChC,mBAAO;AAAA,UACT;AAAA,QACF;AAAA,MACF;AAEA,aAAO;AAAA,IACT;AAAA,IACA,eAAe,qBAAqB,QAAQ,KAAK,UAAU,CAAC;AAAA,IAC5D;AAAA,EACF,CAAC;AAED,SAAO;AACT;AASA,eAAsB,aAAa,KAAU,YAAwD;AACnG,QAAM,OAAO,cAAc,KAAK,UAAU;AAE1C,MAAI;AACF,QAAI,CAAC,MAAM;AACT,aAAO;AAAA,IACT;AAEA,QAAI,KAAK,SAAS;AAChB,aAAO,IAAI,cAAc,aAAa,IAAI;AAAA,IAC5C;AAEA,UAAM,SAAS,KAAK,IAAI;AAExB,UAAM,iBAAiB,IAAI,cAAc,UAAU,KAAK,IAAI;AAC5D,UAAM,aAAa,kBACd,eAAe,UAAU,KAAK,KAAK,SACnC,eAAe,SAAS,KAAK,KAAK,QAClC,IAAI,cAAc,cAAc,eAAe,IAAI;AACxD,QAAI,CAAC,YAAY;AACf,YAAM,IAAI,cAAc,yBAAyB,IAAI;AACrD,YAAM,yBAAyB,GAAG;AAAA,IACpC;AACA,WAAO,IAAI,cAAc,aAAa,IAAI;AAAA,EAC5C,SAAS,OAAO;AACd,QAAI,CAAC,QAAQ,KAAK,SAAS;AACzB,aAAO;AAAA,IACT;AAEA,UAAM;AAAA,EACR;AACF;AAUA,eAAsB,mBAAgD,KAAU,YAAyE;AACvJ,QAAM,QAAQ,MAAM,aAAa,KAAK,UAAU;AAChD,SAAQ,OAAO,eAAe,CAAC;AACjC;AASA,eAAsB,cAAc,KAAU,KAAsC;AAClF,QAAM,UAAU,IAAI,YAAY;AAChC,QAAM,SAAS,QAAQ,OAAO,GAAG,EAAE;AACnC,SAAO,MAAM,IAAI,cAAc,qBAAqB,MAAM,KAAK,CAAC;AAClE;AASO,SAAS,oCAAoC,KAAU,YAAwB,OAA6B;AACjH,QAAM,OAAO,QAAQ,KAAK,YAAY,IAAI;AAC1C,MAAI,CAAC,KAAK,SAAS;AACjB,UAAM,IAAI,MAAM,kBAAkB;AAAA,EACpC;AAEA,MAAI,cAAc,UAAU,KAAK,IAAI,IAAI;AAAA,IACvC,MAAM,KAAK;AAAA,IACX,OAAO;AAAA,IACP,MAAM;AAAA,EACR;AAEA,MAAI,cAAc,cAAc,KAAK,IAAI,IAAI;AAC/C;AAQO,SAAS,cAAc,KAAU,OAA8B;AACpE,QAAM,wBAAwB,yBAAyB,GAAG;AAE1D,WAAS,QAAQ,OAAO;AACtB,WAAO,KAAK,SAAS;AACnB,UAAI,QAAQ,sBAAsB,IAAI,KAAK,IAAI,KAAK;AACpD;AACA,4BAAsB,IAAI,KAAK,MAAM,KAAK;AAE1C,UAAI,MAAM,QAAQ,KAAK,IAAI,IAAI;AAE/B,UAAI,OAAO,IAAI,GAAG;AAChB,YAAI,cAAc,iBAAiB,IAAI,KAAK,KAAK,YAAY,GAAG,IAAI;AAAA,MACtE;AAEA,aAAO,UAAU,KAAK,iBAAiB,KAAK,IAAI,GAAG,IAAI;AAAA,IACzD;AAAA,EACF;AACF;AAWO,SAAS,wBAA2B,KAAU,OAAwB,IAAgB;AAC3F,MAAI;AACF,kBAAc,KAAK,KAAK;AACxB,WAAO,GAAG;AAAA,EACZ,UAAE;AACA,oBAAgB,KAAK,KAAK;AAAA,EAC5B;AACF;AAWA,eAAsB,6BAAgC,KAAU,OAAwB,IAAkC;AACxH,MAAI;AACF,kBAAc,KAAK,KAAK;AACxB,WAAO,MAAM,GAAG;AAAA,EAClB,UAAE;AACA,oBAAgB,KAAK,KAAK;AAAA,EAC5B;AACF;AAQO,SAAS,sCAAsC,KAAU,YAA8B;AAC5F,QAAM,OAAO,QAAQ,KAAK,YAAY,IAAI;AAC1C,MAAI,CAAC,KAAK,SAAS;AACjB,UAAM,IAAI,MAAM,kBAAkB;AAAA,EACpC;AAEA,SAAO,IAAI,cAAc,UAAU,KAAK,IAAI;AAE5C,SAAO,IAAI,cAAc,cAAc,KAAK,IAAI;AAClD;AAQO,SAAS,gBAAgB,KAAU,OAA8B;AACtE,QAAM,wBAAwB,yBAAyB,GAAG;AAE1D,WAAS,QAAQ,OAAO;AACtB,WAAO,KAAK,SAAS;AACnB,UAAI,QAAQ,sBAAsB,IAAI,KAAK,IAAI,KAAK;AACpD;AACA,4BAAsB,IAAI,KAAK,MAAM,KAAK;AAC1C,UAAI,UAAU,GAAG;AACf,8BAAsB,OAAO,KAAK,IAAI;AAEtC,eAAO,IAAI,MAAM,QAAQ,KAAK,IAAI;AAElC,YAAI,OAAO,IAAI,GAAG;AAChB,cAAI,cAAc,iBAAiB,OAAO,KAAK,KAAK,YAAY,GAAG,IAAI;AAAA,QACzE;AAAA,MACF;AAEA,aAAO,UAAU,KAAK,iBAAiB,KAAK,IAAI,GAAG,IAAI;AAAA,IACzD;AAAA,EACF;AACF;AAEA,SAAS,yBAAyB,KAA+B;AAC/D,SAAO,yBAAyB,KAAK,yBAAyB,oBAAI,IAAoB,CAAC,EAAE;AAC3F;",
  "names": []
}

|
|
@@ -5,26 +5,76 @@
|
|
|
5
5
|
*/
|
|
6
6
|
import type { App } from 'obsidian';
|
|
7
7
|
import type { Promisable } from 'type-fest';
|
|
8
|
+
/**
|
|
9
|
+
* Options for the {@link addToQueueAndWait} function.
|
|
10
|
+
*/
|
|
11
|
+
export interface AddToQueueAndWaitOptions {
|
|
12
|
+
/**
|
|
13
|
+
* Optional abort signal.
|
|
14
|
+
*/
|
|
15
|
+
abortSignal?: AbortSignal;
|
|
16
|
+
/**
|
|
17
|
+
* The Obsidian application instance.
|
|
18
|
+
*/
|
|
19
|
+
app: App;
|
|
20
|
+
/**
|
|
21
|
+
* The function to add.
|
|
22
|
+
*/
|
|
23
|
+
operationFn: (abortSignal: AbortSignal) => Promisable<void>;
|
|
24
|
+
/**
|
|
25
|
+
* Optional name of the operation.
|
|
26
|
+
*/
|
|
27
|
+
operationName?: string;
|
|
28
|
+
/**
|
|
29
|
+
* Optional stack trace.
|
|
30
|
+
*/
|
|
31
|
+
stackTrace?: string;
|
|
32
|
+
/**
|
|
33
|
+
* The timeout in milliseconds.
|
|
34
|
+
*/
|
|
35
|
+
timeoutInMilliseconds?: number;
|
|
36
|
+
}
|
|
37
|
+
/**
|
|
38
|
+
* Options for the {@link addToQueue} function.
|
|
39
|
+
*/
|
|
40
|
+
export interface AddToQueueOptions {
|
|
41
|
+
/**
|
|
42
|
+
* Optional abort signal.
|
|
43
|
+
*/
|
|
44
|
+
abortSignal?: AbortSignal;
|
|
45
|
+
/**
|
|
46
|
+
* The Obsidian application instance.
|
|
47
|
+
*/
|
|
48
|
+
app: App;
|
|
49
|
+
/**
|
|
50
|
+
* The function to add.
|
|
51
|
+
*/
|
|
52
|
+
operationFn: (abortSignal: AbortSignal) => Promisable<void>;
|
|
53
|
+
/**
|
|
54
|
+
* Optional name of the operation.
|
|
55
|
+
*/
|
|
56
|
+
operationName?: string;
|
|
57
|
+
/**
|
|
58
|
+
* Optional stack trace.
|
|
59
|
+
*/
|
|
60
|
+
stackTrace?: string;
|
|
61
|
+
/**
|
|
62
|
+
* The timeout in milliseconds.
|
|
63
|
+
*/
|
|
64
|
+
timeoutInMilliseconds?: number;
|
|
65
|
+
}
|
|
8
66
|
/**
|
|
9
67
|
* Adds an asynchronous function to be executed after the previous function completes.
|
|
10
68
|
*
|
|
11
|
-
* @param
|
|
12
|
-
* @param fn - The function to add.
|
|
13
|
-
* @param abortSignal - Optional abort signal.
|
|
14
|
-
* @param timeoutInMilliseconds - The timeout in milliseconds.
|
|
15
|
-
* @param stackTrace - Optional stack trace.
|
|
69
|
+
* @param options - The options for the function.
|
|
16
70
|
*/
|
|
17
|
-
export declare function addToQueue(
|
|
71
|
+
export declare function addToQueue(options: AddToQueueOptions): void;
|
|
18
72
|
/**
|
|
19
73
|
* Adds an asynchronous function to be executed after the previous function completes and returns a {@link Promise} that resolves when the function completes.
|
|
20
74
|
*
|
|
21
|
-
* @param
|
|
22
|
-
* @param fn - The function to add.
|
|
23
|
-
* @param abortSignal - Optional abort signal.
|
|
24
|
-
* @param timeoutInMilliseconds - The timeout in milliseconds.
|
|
25
|
-
* @param stackTrace - Optional stack trace.
|
|
75
|
+
* @param options - The options for the function.
|
|
26
76
|
*/
|
|
27
|
-
export declare function addToQueueAndWait(
|
|
77
|
+
export declare function addToQueueAndWait(options: AddToQueueAndWaitOptions): Promise<void>;
|
|
28
78
|
/**
|
|
29
79
|
* Flushes the queue;
|
|
30
80
|
*
|
|
@@ -25,30 +25,35 @@ import {
|
|
|
25
25
|
} from "../AbortController.mjs";
|
|
26
26
|
import {
|
|
27
27
|
addErrorHandler,
|
|
28
|
-
invokeAsyncSafely
|
|
29
|
-
runWithTimeout
|
|
28
|
+
invokeAsyncSafely
|
|
30
29
|
} from "../Async.mjs";
|
|
31
30
|
import { getStackTrace } from "../Error.mjs";
|
|
32
31
|
import { noop } from "../Function.mjs";
|
|
33
32
|
import { getObsidianDevUtilsState } from "./App.mjs";
|
|
33
|
+
import { runWithTimeoutNotice } from "./AsyncWithNotice.mjs";
|
|
34
34
|
import { invokeAsyncAndLog } from "./Logger.mjs";
|
|
35
|
-
function addToQueue(
|
|
36
|
-
stackTrace
|
|
37
|
-
invokeAsyncSafely(() => addToQueueAndWait(
|
|
35
|
+
function addToQueue(options) {
|
|
36
|
+
const stackTrace = options.stackTrace ?? getStackTrace(1);
|
|
37
|
+
invokeAsyncSafely(() => addToQueueAndWait(options), stackTrace);
|
|
38
38
|
}
|
|
39
|
-
async function addToQueueAndWait(
|
|
40
|
-
abortSignal
|
|
39
|
+
async function addToQueueAndWait(options) {
|
|
40
|
+
const abortSignal = options.abortSignal ?? abortSignalNever();
|
|
41
41
|
abortSignal.throwIfAborted();
|
|
42
42
|
const DEFAULT_TIMEOUT_IN_MILLISECONDS = 6e4;
|
|
43
|
-
timeoutInMilliseconds
|
|
44
|
-
stackTrace
|
|
45
|
-
const
|
|
46
|
-
queue
|
|
47
|
-
queue.
|
|
43
|
+
const timeoutInMilliseconds = options.timeoutInMilliseconds ?? DEFAULT_TIMEOUT_IN_MILLISECONDS;
|
|
44
|
+
const stackTrace = options.stackTrace ?? getStackTrace(1);
|
|
45
|
+
const operationName = options.operationName ?? "";
|
|
46
|
+
const queue = getQueue(options.app).value;
|
|
47
|
+
queue.items.push({ abortSignal, operationFn: options.operationFn, operationName, stackTrace, timeoutInMilliseconds });
|
|
48
|
+
queue.promise = queue.promise.then(() => processNextQueueItem(options.app));
|
|
48
49
|
await queue.promise;
|
|
49
50
|
}
|
|
50
51
|
async function flushQueue(app) {
|
|
51
|
-
await addToQueueAndWait(
|
|
52
|
+
await addToQueueAndWait({
|
|
53
|
+
app,
|
|
54
|
+
operationFn: noop,
|
|
55
|
+
operationName: "Flush queue"
|
|
56
|
+
});
|
|
52
57
|
}
|
|
53
58
|
function getQueue(app) {
|
|
54
59
|
return getObsidianDevUtilsState(app, "queue", { items: [], promise: Promise.resolve() });
|
|
@@ -59,11 +64,22 @@ async function processNextQueueItem(app) {
|
|
|
59
64
|
if (!item) {
|
|
60
65
|
return;
|
|
61
66
|
}
|
|
62
|
-
await addErrorHandler(
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
+
await addErrorHandler(
|
|
68
|
+
() => runWithTimeoutNotice({
|
|
69
|
+
context: { queuedFn: item.operationFn },
|
|
70
|
+
async operationFn(abortSignal) {
|
|
71
|
+
await invokeAsyncAndLog(
|
|
72
|
+
item.operationName || processNextQueueItem.name,
|
|
73
|
+
item.operationFn,
|
|
74
|
+
abortSignalAny(abortSignal, item.abortSignal),
|
|
75
|
+
item.stackTrace
|
|
76
|
+
);
|
|
77
|
+
},
|
|
78
|
+
operationName: item.operationName,
|
|
79
|
+
stackTrace: item.stackTrace,
|
|
80
|
+
timeoutInMilliseconds: item.timeoutInMilliseconds
|
|
81
|
+
})
|
|
82
|
+
);
|
|
67
83
|
queue.items.shift();
|
|
68
84
|
}
|
|
69
85
|
export {
|
|
@@ -71,4 +87,4 @@ export {
|
|
|
71
87
|
addToQueueAndWait,
|
|
72
88
|
flushQueue
|
|
73
89
|
};
|
|
74
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
90
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Queue.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility functions for enqueuing and processing functions in Obsidian.\n */\n\nimport type { App } from 'obsidian';\nimport type { Promisable } from 'type-fest';\n\nimport type { ValueWrapper } from './App.ts';\n\nimport {\n  abortSignalAny,\n  abortSignalNever\n} from '../AbortController.ts';\nimport {\n  addErrorHandler,\n  invokeAsyncSafely\n} from '../Async.ts';\nimport { getStackTrace } from '../Error.ts';\nimport { noop } from '../Function.ts';\nimport { getObsidianDevUtilsState } from './App.ts';\nimport { runWithTimeoutNotice } from './AsyncWithNotice.ts';\nimport { invokeAsyncAndLog } from './Logger.ts';\n\n/**\n * Options for the {@link addToQueueAndWait} function.\n */\nexport interface AddToQueueAndWaitOptions {\n  /**\n   * Optional abort signal.\n   */\n  abortSignal?: AbortSignal;\n\n  /**\n   * The Obsidian application instance.\n   */\n  app: App;\n\n  /**\n   * The function to add.\n   */\n  operationFn: (abortSignal: AbortSignal) => Promisable<void>;\n\n  /**\n   * Optional name of the operation.\n   */\n  operationName?: string;\n\n  /**\n   * Optional stack trace.\n   */\n  stackTrace?: string;\n\n  /**\n   * The timeout in milliseconds.\n   */\n  timeoutInMilliseconds?: number;\n}\n\n/**\n * Options for the {@link addToQueue} function.\n */\nexport interface AddToQueueOptions {\n  /**\n   * Optional abort signal.\n   */\n  abortSignal?: AbortSignal;\n\n  /**\n   * The Obsidian application instance.\n   */\n  app: App;\n\n  /**\n   * The function to add.\n   */\n  operationFn: (abortSignal: AbortSignal) => Promisable<void>;\n\n  /**\n   * Optional name of the operation.\n   */\n  operationName?: string;\n\n  /**\n   * Optional stack trace.\n   */\n  stackTrace?: string;\n\n  /**\n   * The timeout in milliseconds.\n   */\n  timeoutInMilliseconds?: number;\n}\n\ninterface Queue {\n  items: QueueItem[];\n  promise: Promise<void>;\n}\n\ninterface QueueItem {\n  abortSignal: AbortSignal;\n  operationFn(this: void, abortSignal: AbortSignal): Promisable<void>;\n  operationName: string;\n  stackTrace: string;\n  timeoutInMilliseconds: number;\n}\n\n/**\n * Adds an asynchronous function to be executed after the previous function completes.\n *\n * @param options - The options for the function.\n */\nexport function addToQueue(options: AddToQueueOptions): void {\n  const stackTrace = options.stackTrace ?? getStackTrace(1);\n  invokeAsyncSafely(() => addToQueueAndWait(options), stackTrace);\n}\n\n/**\n * Adds an asynchronous function to be executed after the previous function completes and returns a {@link Promise} that resolves when the function completes.\n *\n * @param options - The options for the function.\n */\nexport async function addToQueueAndWait(options: AddToQueueAndWaitOptions): Promise<void> {\n  const abortSignal = options.abortSignal ?? abortSignalNever();\n  abortSignal.throwIfAborted();\n\n  const DEFAULT_TIMEOUT_IN_MILLISECONDS = 60000;\n  const timeoutInMilliseconds = options.timeoutInMilliseconds ?? DEFAULT_TIMEOUT_IN_MILLISECONDS;\n  const stackTrace = options.stackTrace ?? getStackTrace(1);\n  const operationName = options.operationName ?? '';\n  const queue = getQueue(options.app).value;\n  queue.items.push({ abortSignal, operationFn: options.operationFn, operationName, stackTrace, timeoutInMilliseconds });\n  queue.promise = queue.promise.then(() => processNextQueueItem(options.app));\n  await queue.promise;\n}\n\n/**\n * Flushes the queue;\n *\n * @param app - The Obsidian application instance.\n */\nexport async function flushQueue(app: App): Promise<void> {\n  await addToQueueAndWait({\n    app,\n    operationFn: noop,\n    operationName: 'Flush queue'\n  });\n}\n\nfunction getQueue(app: App): ValueWrapper<Queue> {\n  return getObsidianDevUtilsState(app, 'queue', { items: [], promise: Promise.resolve() });\n}\n\nasync function processNextQueueItem(app: App): Promise<void> {\n  const queue = getQueue(app).value;\n  const item = queue.items[0];\n  if (!item) {\n    return;\n  }\n\n  await addErrorHandler(() =>\n    runWithTimeoutNotice({\n      context: { queuedFn: item.operationFn },\n      async operationFn(abortSignal: AbortSignal): Promise<void> {\n        await invokeAsyncAndLog(\n          item.operationName || processNextQueueItem.name,\n          item.operationFn,\n          abortSignalAny(abortSignal, item.abortSignal),\n          item.stackTrace\n        );\n      },\n      operationName: item.operationName,\n      stackTrace: item.stackTrace,\n      timeoutInMilliseconds: item.timeoutInMilliseconds\n    })\n  );\n  queue.items.shift();\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;AAWA;AAAA,EACE;AAAA,EACA;AAAA,OACK;AACP;AAAA,EACE;AAAA,EACA;AAAA,OACK;AACP,SAAS,qBAAqB;AAC9B,SAAS,YAAY;AACrB,SAAS,gCAAgC;AACzC,SAAS,4BAA4B;AACrC,SAAS,yBAAyB;AA0F3B,SAAS,WAAW,SAAkC;AAC3D,QAAM,aAAa,QAAQ,cAAc,cAAc,CAAC;AACxD,oBAAkB,MAAM,kBAAkB,OAAO,GAAG,UAAU;AAChE;AAOA,eAAsB,kBAAkB,SAAkD;AACxF,QAAM,cAAc,QAAQ,eAAe,iBAAiB;AAC5D,cAAY,eAAe;AAE3B,QAAM,kCAAkC;AACxC,QAAM,wBAAwB,QAAQ,yBAAyB;AAC/D,QAAM,aAAa,QAAQ,cAAc,cAAc,CAAC;AACxD,QAAM,gBAAgB,QAAQ,iBAAiB;AAC/C,QAAM,QAAQ,SAAS,QAAQ,GAAG,EAAE;AACpC,QAAM,MAAM,KAAK,EAAE,aAAa,aAAa,QAAQ,aAAa,eAAe,YAAY,sBAAsB,CAAC;AACpH,QAAM,UAAU,MAAM,QAAQ,KAAK,MAAM,qBAAqB,QAAQ,GAAG,CAAC;AAC1E,QAAM,MAAM;AACd;AAOA,eAAsB,WAAW,KAAyB;AACxD,QAAM,kBAAkB;AAAA,IACtB;AAAA,IACA,aAAa;AAAA,IACb,eAAe;AAAA,EACjB,CAAC;AACH;AAEA,SAAS,SAAS,KAA+B;AAC/C,SAAO,yBAAyB,KAAK,SAAS,EAAE,OAAO,CAAC,GAAG,SAAS,QAAQ,QAAQ,EAAE,CAAC;AACzF;AAEA,eAAe,qBAAqB,KAAyB;AAC3D,QAAM,QAAQ,SAAS,GAAG,EAAE;AAC5B,QAAM,OAAO,MAAM,MAAM,CAAC;AAC1B,MAAI,CAAC,MAAM;AACT;AAAA,EACF;AAEA,QAAM;AAAA,IAAgB,MACpB,qBAAqB;AAAA,MACnB,SAAS,EAAE,UAAU,KAAK,YAAY;AAAA,MACtC,MAAM,YAAY,aAAyC;AACzD,cAAM;AAAA,UACJ,KAAK,iBAAiB,qBAAqB;AAAA,UAC3C,KAAK;AAAA,UACL,eAAe,aAAa,KAAK,WAAW;AAAA,UAC5C,KAAK;AAAA,QACP;AAAA,MACF;AAAA,MACA,eAAe,KAAK;AAAA,MACpB,YAAY,KAAK;AAAA,MACjB,uBAAuB,KAAK;AAAA,IAC9B,CAAC;AAAA,EACH;AACA,QAAM,MAAM,MAAM;AACpB;",
  "names": []
}

|