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
|
@@ -0,0 +1,167 @@
|
|
|
1
|
+
/*
|
|
2
|
+
THIS IS A GENERATED/BUNDLED FILE BY ESBUILD
|
|
3
|
+
if you want to view the source, please visit the github repository of this plugin
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
(function initCjs() {
|
|
7
|
+
const globalThisRecord = globalThis;
|
|
8
|
+
globalThisRecord['__name'] ??= name;
|
|
9
|
+
const originalRequire = require;
|
|
10
|
+
if (originalRequire && !originalRequire.__isPatched) {
|
|
11
|
+
// eslint-disable-next-line no-global-assign, no-implicit-globals -- We need to patch the `require()` function.
|
|
12
|
+
require = Object.assign(
|
|
13
|
+
(id) => requirePatched(id),
|
|
14
|
+
originalRequire,
|
|
15
|
+
{
|
|
16
|
+
__isPatched: true
|
|
17
|
+
}
|
|
18
|
+
);
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
const newFuncs = {
|
|
22
|
+
__extractDefault() {
|
|
23
|
+
return extractDefault;
|
|
24
|
+
},
|
|
25
|
+
process() {
|
|
26
|
+
const browserProcess = {
|
|
27
|
+
browser: true,
|
|
28
|
+
cwd() {
|
|
29
|
+
return '/';
|
|
30
|
+
},
|
|
31
|
+
env: {},
|
|
32
|
+
platform: 'android'
|
|
33
|
+
};
|
|
34
|
+
return browserProcess;
|
|
35
|
+
}
|
|
36
|
+
};
|
|
37
|
+
|
|
38
|
+
for (const key of Object.keys(newFuncs)) {
|
|
39
|
+
globalThisRecord[key] ??= newFuncs[key]?.();
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
function name(obj) {
|
|
43
|
+
return obj;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
function extractDefault(module) {
|
|
47
|
+
return module && module.__esModule && 'default' in module ? module.default : module;
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
const OBSIDIAN_BUILT_IN_MODULE_NAMES = [
|
|
51
|
+
'obsidian',
|
|
52
|
+
'@codemirror/autocomplete',
|
|
53
|
+
'@codemirror/collab',
|
|
54
|
+
'@codemirror/commands',
|
|
55
|
+
'@codemirror/language',
|
|
56
|
+
'@codemirror/lint',
|
|
57
|
+
'@codemirror/search',
|
|
58
|
+
'@codemirror/state',
|
|
59
|
+
'@codemirror/text',
|
|
60
|
+
'@codemirror/view',
|
|
61
|
+
'@lezer/common',
|
|
62
|
+
'@lezer/lr',
|
|
63
|
+
'@lezer/highlight'];
|
|
64
|
+
|
|
65
|
+
|
|
66
|
+
const DEPRECATED_OBSIDIAN_BUILT_IN_MODULE_NAMES = [
|
|
67
|
+
'@codemirror/closebrackets',
|
|
68
|
+
'@codemirror/comment',
|
|
69
|
+
'@codemirror/fold',
|
|
70
|
+
'@codemirror/gutter',
|
|
71
|
+
'@codemirror/highlight',
|
|
72
|
+
'@codemirror/history',
|
|
73
|
+
'@codemirror/matchbrackets',
|
|
74
|
+
'@codemirror/panel',
|
|
75
|
+
'@codemirror/rangeset',
|
|
76
|
+
'@codemirror/rectangular-selection',
|
|
77
|
+
'@codemirror/stream-parser',
|
|
78
|
+
'@codemirror/tooltip'];
|
|
79
|
+
|
|
80
|
+
|
|
81
|
+
function requirePatched(id) {
|
|
82
|
+
if (OBSIDIAN_BUILT_IN_MODULE_NAMES.includes(id) || DEPRECATED_OBSIDIAN_BUILT_IN_MODULE_NAMES.includes(id)) {
|
|
83
|
+
return originalRequire?.(id);
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
// eslint-disable-next-line @typescript-eslint/no-deprecated, @typescript-eslint/no-unnecessary-condition -- We need access to app here which might not be available yet.
|
|
87
|
+
if (globalThis?.app?.isMobile) {
|
|
88
|
+
if (id === 'process' || id === 'node:process') {
|
|
89
|
+
console.debug(`The most likely you can safely ignore this error. Module not found: ${id}. Fake process object is returned instead.`);
|
|
90
|
+
return globalThis.process;
|
|
91
|
+
}
|
|
92
|
+
} else {
|
|
93
|
+
const module = originalRequire?.(id);
|
|
94
|
+
if (module) {
|
|
95
|
+
return extractDefault(module);
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
console.debug(`The most likely you can safely ignore this error. Module not found: ${id}. Empty object is returned instead.`);
|
|
100
|
+
return {};
|
|
101
|
+
}
|
|
102
|
+
})();
|
|
103
|
+
|
|
104
|
+
"use strict";
|
|
105
|
+
var __defProp = Object.defineProperty;
|
|
106
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
107
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
108
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
109
|
+
var __export = (target, all) => {
|
|
110
|
+
for (var name in all)
|
|
111
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
|
112
|
+
};
|
|
113
|
+
var __copyProps = (to, from, except, desc) => {
|
|
114
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
115
|
+
for (let key of __getOwnPropNames(from))
|
|
116
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
117
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
118
|
+
}
|
|
119
|
+
return to;
|
|
120
|
+
};
|
|
121
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
122
|
+
var AsyncWithNotice_exports = {};
|
|
123
|
+
__export(AsyncWithNotice_exports, {
|
|
124
|
+
retryWithTimeoutNotice: () => retryWithTimeoutNotice,
|
|
125
|
+
runWithTimeoutNotice: () => runWithTimeoutNotice
|
|
126
|
+
});
|
|
127
|
+
module.exports = __toCommonJS(AsyncWithNotice_exports);
|
|
128
|
+
var import_obsidian = require('obsidian');
|
|
129
|
+
var import_Async = require('../Async.cjs');
|
|
130
|
+
async function retryWithTimeoutNotice(options) {
|
|
131
|
+
return (0, import_Async.retryWithTimeout)({
|
|
132
|
+
...options,
|
|
133
|
+
onTimeout: onTimeoutNotice
|
|
134
|
+
});
|
|
135
|
+
}
|
|
136
|
+
async function runWithTimeoutNotice(options) {
|
|
137
|
+
return (0, import_Async.runWithTimeout)({
|
|
138
|
+
...options,
|
|
139
|
+
onTimeout: onTimeoutNotice
|
|
140
|
+
});
|
|
141
|
+
}
|
|
142
|
+
function onTimeoutNotice(ctx) {
|
|
143
|
+
const notice = new import_obsidian.Notice(createFragment((f) => {
|
|
144
|
+
if (ctx.operationName) {
|
|
145
|
+
f.appendText(`Operation: ${ctx.operationName}`);
|
|
146
|
+
f.createEl("br");
|
|
147
|
+
}
|
|
148
|
+
f.appendText(`The operation timed out after ${String(ctx.duration)} milliseconds.`);
|
|
149
|
+
f.createEl("br");
|
|
150
|
+
const button = f.createEl("button", {
|
|
151
|
+
text: "Cancel"
|
|
152
|
+
});
|
|
153
|
+
button.addEventListener("click", () => {
|
|
154
|
+
ctx.terminateOperation();
|
|
155
|
+
notice.hide();
|
|
156
|
+
});
|
|
157
|
+
}));
|
|
158
|
+
ctx.onOperationCompleted(() => {
|
|
159
|
+
notice.hide();
|
|
160
|
+
});
|
|
161
|
+
}
|
|
162
|
+
// Annotate the CommonJS export names for ESM import in node:
|
|
163
|
+
0 && (module.exports = {
|
|
164
|
+
retryWithTimeoutNotice,
|
|
165
|
+
runWithTimeoutNotice
|
|
166
|
+
});
|
|
167
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vc3JjL29ic2lkaWFuL0FzeW5jV2l0aE5vdGljZS50cyJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiLyoqXG4gKiBAcGFja2FnZURvY3VtZW50YXRpb25cbiAqXG4gKiBQcm92aWRlcyBhIHV0aWxpdHkgdG8gZXhlY3V0ZSBhbiBhc3luY2hyb25vdXMgZnVuY3Rpb24gd2l0aCBhIG5vdGljZS5cbiAqL1xuXG5pbXBvcnQgdHlwZSB7IFByb21pc2FibGUgfSBmcm9tICd0eXBlLWZlc3QnO1xuXG5pbXBvcnQgeyBOb3RpY2UgfSBmcm9tICdvYnNpZGlhbic7XG5cbmltcG9ydCB0eXBlIHtcbiAgUmV0cnlPcHRpb25zLFxuICBUaW1lb3V0Q29udGV4dFxufSBmcm9tICcuLi9Bc3luYy50cyc7XG5cbmltcG9ydCB7XG4gIHJldHJ5V2l0aFRpbWVvdXQsXG4gIHJ1bldpdGhUaW1lb3V0XG59IGZyb20gJy4uL0FzeW5jLnRzJztcblxuLyoqXG4gKiBPcHRpb25zIGZvciB7QGxpbmsgcmV0cnlXaXRoVGltZW91dE5vdGljZX0uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgUmV0cnlXaXRoVGltZW91dE5vdGljZU9wdGlvbnMge1xuICAvKipcbiAgICogVGhlIG9wZXJhdGlvbiBmdW5jdGlvbiB0byBleGVjdXRlLlxuICAgKlxuICAgKiBAcGFyYW0gYWJvcnRTaWduYWwgLSBUaGUgYWJvcnQgc2lnbmFsIHRvIGxpc3RlbiB0by5cbiAgICogQHJldHVybnMgVGhlIHJlc3VsdCBvZiB0aGUgZnVuY3Rpb24uXG4gICAqL1xuICBvcGVyYXRpb25Gbih0aGlzOiB2b2lkLCBhYm9ydFNpZ25hbDogQWJvcnRTaWduYWwpOiBQcm9taXNhYmxlPGJvb2xlYW4+O1xuXG4gIC8qKlxuICAgKiBUaGUgbmFtZSBvZiB0aGUgb3BlcmF0aW9uLlxuICAgKi9cbiAgb3BlcmF0aW9uTmFtZT86IHN0cmluZztcblxuICAvKipcbiAgICogVGhlIHJldHJ5IG9wdGlvbnMuXG4gICAqL1xuICByZXRyeU9wdGlvbnM/OiBSZXRyeU9wdGlvbnM7XG5cbiAgLyoqXG4gICAqIFRoZSBzdGFjayB0cmFjZSBvZiB0aGUgc291cmNlIGZ1bmN0aW9uLlxuICAgKi9cbiAgc3RhY2tUcmFjZT86IHN0cmluZztcbn1cblxuLyoqXG4gKiBPcHRpb25zIGZvciB7QGxpbmsgcnVuV2l0aFRpbWVvdXR9LlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFJ1bldpdGhUaW1lb3V0Tm90aWNlT3B0aW9uczxSZXN1bHQ+IHtcbiAgLyoqXG4gICAqIFRoZSBjb250ZXh0IG9mIHRoZSBmdW5jdGlvbi5cbiAgICovXG4gIGNvbnRleHQ/OiB1bmtub3duO1xuICAvKipcbiAgICogVGhlIG9wZXJhdGlvbiBmdW5jdGlvbiB0byBleGVjdXRlLlxuICAgKlxuICAgKiBAcGFyYW0gYWJvcnRTaWduYWwgLSBUaGUgYWJvcnQgc2lnbmFsIHRvIGxpc3RlbiB0by5cbiAgICogQHJldHVybnMgVGhlIHJlc3VsdCBvZiB0aGUgZnVuY3Rpb24uXG4gICAqL1xuICBvcGVyYXRpb25GbihhYm9ydFNpZ25hbDogQWJvcnRTaWduYWwpOiBQcm9taXNhYmxlPFJlc3VsdD47XG4gIC8qKlxuICAgKiBUaGUgbmFtZSBvZiB0aGUgb3BlcmF0aW9uLlxuICAgKi9cbiAgb3BlcmF0aW9uTmFtZT86IHN0cmluZztcbiAgLyoqXG4gICAqIFRoZSBzdGFjayB0cmFjZSBvZiB0aGUgc291cmNlIGZ1bmN0aW9uLlxuICAgKi9cbiAgc3RhY2tUcmFjZT86IHN0cmluZztcblxuICAvKipcbiAgICogVGhlIG1heGltdW0gdGltZSB0byB3YWl0IGluIG1pbGxpc2Vjb25kcy5cbiAgICovXG4gIHRpbWVvdXRJbk1pbGxpc2Vjb25kczogbnVtYmVyO1xufVxuXG4vKipcbiAqIFJldHJpZXMgdGhlIHByb3ZpZGVkIGZ1bmN0aW9uIHVudGlsIGl0IHJldHVybnMgdHJ1ZSBvciB0aGUgdGltZW91dCBpcyByZWFjaGVkIGFuZCBkaXNwbGF5cyBhIG5vdGljZSBpZiB0aGUgZnVuY3Rpb24gdGltZXMgb3V0LlxuICpcbiAqIEBwYXJhbSBvcHRpb25zIC0gVGhlIG9wdGlvbnMgZm9yIHRoZSBmdW5jdGlvbi5cbiAqIEByZXR1cm5zIEEge0BsaW5rIFByb21pc2V9IHRoYXQgcmVzb2x2ZXMgd2hlbiB0aGUgZnVuY3Rpb24gcmV0dXJucyB0cnVlIG9yIHJlamVjdHMgd2hlbiB0aGUgdGltZW91dCBpcyByZWFjaGVkLlxuICovXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gcmV0cnlXaXRoVGltZW91dE5vdGljZShvcHRpb25zOiBSZXRyeVdpdGhUaW1lb3V0Tm90aWNlT3B0aW9ucyk6IFByb21pc2U8dm9pZD4ge1xuICByZXR1cm4gcmV0cnlXaXRoVGltZW91dCh7XG4gICAgLi4ub3B0aW9ucyxcbiAgICBvblRpbWVvdXQ6IG9uVGltZW91dE5vdGljZVxuICB9KTtcbn1cblxuLyoqXG4gKiBFeGVjdXRlcyBhIGZ1bmN0aW9uIHdpdGggYSB0aW1lb3V0IGFuZCBkaXNwbGF5cyBhIG5vdGljZSBpZiB0aGUgZnVuY3Rpb24gdGltZXMgb3V0LlxuICpcbiAqIEB0eXBlUGFyYW0gUiAtIFRoZSB0eXBlIG9mIHRoZSByZXN1bHQgZnJvbSB0aGUgYXN5bmNocm9ub3VzIGZ1bmN0aW9uLlxuICogQHBhcmFtIG9wdGlvbnMgLSBUaGUgb3B0aW9ucyBmb3IgdGhlIGZ1bmN0aW9uLlxuICogQHJldHVybnMgVGhlIHJlc3VsdCBvZiB0aGUgZnVuY3Rpb24uXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBydW5XaXRoVGltZW91dE5vdGljZTxSZXN1bHQ+KG9wdGlvbnM6IFJ1bldpdGhUaW1lb3V0Tm90aWNlT3B0aW9uczxSZXN1bHQ+KTogUHJvbWlzZTxSZXN1bHQ+IHtcbiAgcmV0dXJuIHJ1bldpdGhUaW1lb3V0KHtcbiAgICAuLi5vcHRpb25zLFxuICAgIG9uVGltZW91dDogb25UaW1lb3V0Tm90aWNlXG4gIH0pO1xufVxuXG5mdW5jdGlvbiBvblRpbWVvdXROb3RpY2UoY3R4OiBUaW1lb3V0Q29udGV4dCk6IHZvaWQge1xuICBjb25zdCBub3RpY2UgPSBuZXcgTm90aWNlKGNyZWF0ZUZyYWdtZW50KChmKSA9PiB7XG4gICAgaWYgKGN0eC5vcGVyYXRpb25OYW1lKSB7XG4gICAgICBmLmFwcGVuZFRleHQoYE9wZXJhdGlvbjogJHtjdHgub3BlcmF0aW9uTmFtZX1gKTtcbiAgICAgIGYuY3JlYXRlRWwoJ2JyJyk7XG4gICAgfVxuICAgIGYuYXBwZW5kVGV4dChgVGhlIG9wZXJhdGlvbiB0aW1lZCBvdXQgYWZ0ZXIgJHtTdHJpbmcoY3R4LmR1cmF0aW9uKX0gbWlsbGlzZWNvbmRzLmApO1xuICAgIGYuY3JlYXRlRWwoJ2JyJyk7XG4gICAgY29uc3QgYnV0dG9uID0gZi5jcmVhdGVFbCgnYnV0dG9uJywge1xuICAgICAgdGV4dDogJ0NhbmNlbCdcbiAgICB9KTtcbiAgICBidXR0b24uYWRkRXZlbnRMaXN0ZW5lcignY2xpY2snLCAoKSA9PiB7XG4gICAgICBjdHgudGVybWluYXRlT3BlcmF0aW9uKCk7XG4gICAgICBub3RpY2UuaGlkZSgpO1xuICAgIH0pO1xuICB9KSk7XG5cbiAgY3R4Lm9uT3BlcmF0aW9uQ29tcGxldGVkKCgpID0+IHtcbiAgICBub3RpY2UuaGlkZSgpO1xuICB9KTtcbn1cbiJdLAogICJtYXBwaW5ncyI6ICI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFRQSxzQkFBdUI7QUFPdkIsbUJBR087QUFrRVAsZUFBc0IsdUJBQXVCLFNBQXVEO0FBQ2xHLGFBQU8sK0JBQWlCO0FBQUEsSUFDdEIsR0FBRztBQUFBLElBQ0gsV0FBVztBQUFBLEVBQ2IsQ0FBQztBQUNIO0FBU0EsZUFBc0IscUJBQTZCLFNBQStEO0FBQ2hILGFBQU8sNkJBQWU7QUFBQSxJQUNwQixHQUFHO0FBQUEsSUFDSCxXQUFXO0FBQUEsRUFDYixDQUFDO0FBQ0g7QUFFQSxTQUFTLGdCQUFnQixLQUEyQjtBQUNsRCxRQUFNLFNBQVMsSUFBSSx1QkFBTyxlQUFlLENBQUMsTUFBTTtBQUM5QyxRQUFJLElBQUksZUFBZTtBQUNyQixRQUFFLFdBQVcsY0FBYyxJQUFJLGFBQWEsRUFBRTtBQUM5QyxRQUFFLFNBQVMsSUFBSTtBQUFBLElBQ2pCO0FBQ0EsTUFBRSxXQUFXLGlDQUFpQyxPQUFPLElBQUksUUFBUSxDQUFDLGdCQUFnQjtBQUNsRixNQUFFLFNBQVMsSUFBSTtBQUNmLFVBQU0sU0FBUyxFQUFFLFNBQVMsVUFBVTtBQUFBLE1BQ2xDLE1BQU07QUFBQSxJQUNSLENBQUM7QUFDRCxXQUFPLGlCQUFpQixTQUFTLE1BQU07QUFDckMsVUFBSSxtQkFBbUI7QUFDdkIsYUFBTyxLQUFLO0FBQUEsSUFDZCxDQUFDO0FBQUEsRUFDSCxDQUFDLENBQUM7QUFFRixNQUFJLHFCQUFxQixNQUFNO0FBQzdCLFdBQU8sS0FBSztBQUFBLEVBQ2QsQ0FBQztBQUNIOyIsCiAgIm5hbWVzIjogW10KfQo=
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @packageDocumentation
|
|
3
|
+
*
|
|
4
|
+
* Provides a utility to execute an asynchronous function with a notice.
|
|
5
|
+
*/
|
|
6
|
+
import type { Promisable } from 'type-fest';
|
|
7
|
+
import type { RetryOptions } from '../Async.cjs';
|
|
8
|
+
/**
|
|
9
|
+
* Options for {@link retryWithTimeoutNotice}.
|
|
10
|
+
*/
|
|
11
|
+
export interface RetryWithTimeoutNoticeOptions {
|
|
12
|
+
/**
|
|
13
|
+
* The operation function to execute.
|
|
14
|
+
*
|
|
15
|
+
* @param abortSignal - The abort signal to listen to.
|
|
16
|
+
* @returns The result of the function.
|
|
17
|
+
*/
|
|
18
|
+
operationFn(this: void, abortSignal: AbortSignal): Promisable<boolean>;
|
|
19
|
+
/**
|
|
20
|
+
* The name of the operation.
|
|
21
|
+
*/
|
|
22
|
+
operationName?: string;
|
|
23
|
+
/**
|
|
24
|
+
* The retry options.
|
|
25
|
+
*/
|
|
26
|
+
retryOptions?: RetryOptions;
|
|
27
|
+
/**
|
|
28
|
+
* The stack trace of the source function.
|
|
29
|
+
*/
|
|
30
|
+
stackTrace?: string;
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* Options for {@link runWithTimeout}.
|
|
34
|
+
*/
|
|
35
|
+
export interface RunWithTimeoutNoticeOptions<Result> {
|
|
36
|
+
/**
|
|
37
|
+
* The context of the function.
|
|
38
|
+
*/
|
|
39
|
+
context?: unknown;
|
|
40
|
+
/**
|
|
41
|
+
* The operation function to execute.
|
|
42
|
+
*
|
|
43
|
+
* @param abortSignal - The abort signal to listen to.
|
|
44
|
+
* @returns The result of the function.
|
|
45
|
+
*/
|
|
46
|
+
operationFn(abortSignal: AbortSignal): Promisable<Result>;
|
|
47
|
+
/**
|
|
48
|
+
* The name of the operation.
|
|
49
|
+
*/
|
|
50
|
+
operationName?: string;
|
|
51
|
+
/**
|
|
52
|
+
* The stack trace of the source function.
|
|
53
|
+
*/
|
|
54
|
+
stackTrace?: string;
|
|
55
|
+
/**
|
|
56
|
+
* The maximum time to wait in milliseconds.
|
|
57
|
+
*/
|
|
58
|
+
timeoutInMilliseconds: number;
|
|
59
|
+
}
|
|
60
|
+
/**
|
|
61
|
+
* Retries the provided function until it returns true or the timeout is reached and displays a notice if the function times out.
|
|
62
|
+
*
|
|
63
|
+
* @param options - The options for the function.
|
|
64
|
+
* @returns A {@link Promise} that resolves when the function returns true or rejects when the timeout is reached.
|
|
65
|
+
*/
|
|
66
|
+
export declare function retryWithTimeoutNotice(options: RetryWithTimeoutNoticeOptions): Promise<void>;
|
|
67
|
+
/**
|
|
68
|
+
* Executes a function with a timeout and displays a notice if the function times out.
|
|
69
|
+
*
|
|
70
|
+
* @typeParam R - The type of the result from the asynchronous function.
|
|
71
|
+
* @param options - The options for the function.
|
|
72
|
+
* @returns The result of the function.
|
|
73
|
+
*/
|
|
74
|
+
export declare function runWithTimeoutNotice<Result>(options: RunWithTimeoutNoticeOptions<Result>): Promise<Result>;
|
|
@@ -127,6 +127,7 @@ __export(Callout_exports, {
|
|
|
127
127
|
});
|
|
128
128
|
module.exports = __toCommonJS(Callout_exports);
|
|
129
129
|
var import_Error = require('../Error.cjs');
|
|
130
|
+
var import_ObjectUtils = require('../ObjectUtils.cjs');
|
|
130
131
|
var import_ValueProvider = require('../ValueProvider.cjs');
|
|
131
132
|
var import_Dataview = require('./Dataview.cjs');
|
|
132
133
|
var import_Queue = require('./Queue.cjs');
|
|
@@ -154,7 +155,12 @@ function renderCallout(options) {
|
|
|
154
155
|
for (const entry of entries) {
|
|
155
156
|
if (entry.isIntersecting) {
|
|
156
157
|
observer.unobserve(entry.target);
|
|
157
|
-
(0, import_Queue.addToQueue)(
|
|
158
|
+
(0, import_Queue.addToQueue)((0, import_ObjectUtils.normalizeOptionalProperties)({
|
|
159
|
+
abortSignal: options.abortSignal,
|
|
160
|
+
app: dv.app,
|
|
161
|
+
operationFn: loadContent,
|
|
162
|
+
operationName: "Load content for callout"
|
|
163
|
+
}));
|
|
158
164
|
}
|
|
159
165
|
}
|
|
160
166
|
});
|
|
@@ -193,4 +199,4 @@ function getModifier(mode) {
|
|
|
193
199
|
renderCallout,
|
|
194
200
|
wrapForCallout
|
|
195
201
|
});
|
|
196
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
202
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vc3JjL29ic2lkaWFuL0NhbGxvdXQudHMiXSwKICAic291cmNlc0NvbnRlbnQiOiBbIi8qKlxuICogQHBhY2thZ2VEb2N1bWVudGF0aW9uXG4gKlxuICogVGhpcyBtb2R1bGUgcHJvdmlkZXMgdXRpbGl0eSBmdW5jdGlvbnMgZm9yIHJlbmRlcmluZyBjYWxsb3V0cyBpbiBEYXRhdmlldy5cbiAqL1xuXG5pbXBvcnQgdHlwZSB7IE1heWJlUmV0dXJuIH0gZnJvbSAnLi4vVHlwZS50cyc7XG5pbXBvcnQgdHlwZSB7IFZhbHVlUHJvdmlkZXIgfSBmcm9tICcuLi9WYWx1ZVByb3ZpZGVyLnRzJztcbmltcG9ydCB0eXBlIHsgRGF0YXZpZXdJbmxpbmVBcGkgfSBmcm9tICcuL0RhdGF2aWV3LnRzJztcbmltcG9ydCB0eXBlIHsgQWRkVG9RdWV1ZU9wdGlvbnMgfSBmcm9tICcuL1F1ZXVlLnRzJztcblxuaW1wb3J0IHsgdGhyb3dFeHByZXNzaW9uIH0gZnJvbSAnLi4vRXJyb3IudHMnO1xuaW1wb3J0IHsgbm9ybWFsaXplT3B0aW9uYWxQcm9wZXJ0aWVzIH0gZnJvbSAnLi4vT2JqZWN0VXRpbHMudHMnO1xuaW1wb3J0IHsgcmVzb2x2ZVZhbHVlIH0gZnJvbSAnLi4vVmFsdWVQcm92aWRlci50cyc7XG5pbXBvcnQgeyBnZXRSZW5kZXJlZENvbnRhaW5lciB9IGZyb20gJy4vRGF0YXZpZXcudHMnO1xuaW1wb3J0IHsgYWRkVG9RdWV1ZSB9IGZyb20gJy4vUXVldWUudHMnO1xuXG4vKipcbiAqIEVudW0gcmVwcmVzZW50aW5nIHRoZSBtb2RlIG9mIGEgY2FsbG91dC5cbiAqL1xuZXhwb3J0IGVudW0gQ2FsbG91dE1vZGUge1xuICAvKiogRGVmYXVsdCBtb2RlLCB3aXRoIG5vIHNwZWNpYWwgYmVoYXZpb3IuICovXG4gIERlZmF1bHQsXG5cbiAgLyoqIEZvbGRhYmxlIG1vZGUgd2l0aCB0aGUgY2FsbG91dCBjb2xsYXBzZWQuICovXG4gIEZvbGRhYmxlQ29sbGFwc2VkLFxuXG4gIC8qKiBGb2xkYWJsZSBtb2RlIHdpdGggdGhlIGNhbGxvdXQgZXhwYW5kZWQuICovXG4gIEZvbGRhYmxlRXhwYW5kZWRcbn1cblxuLyoqXG4gKiBPcHRpb25zIGZvciB7QGxpbmsgcmVuZGVyQ2FsbG91dH0uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgUmVuZGVyQ2FsbG91dE9wdGlvbnMge1xuICAvKipcbiAgICogQW4gYWJvcnQgc2lnbmFsLlxuICAgKi9cbiAgYWJvcnRTaWduYWw/OiBBYm9ydFNpZ25hbDtcblxuICAvKipcbiAgICogQW4gb3B0aW9uYWwgcHJvdmlkZXIgZm9yIHRoZSBjb250ZW50LCB3aGljaCBjYW4gYmUgZWl0aGVyIGEgc3RyaW5nIG9yIGEgTm9kZS5cbiAgICovXG4gIGNvbnRlbnRQcm92aWRlcj86IFZhbHVlUHJvdmlkZXI8TWF5YmVSZXR1cm48Tm9kZSB8IHN0cmluZz4+O1xuXG4gIC8qKlxuICAgKiBBIHtAbGluayBEYXRhdmlld0lubGluZUFwaX0gaW5zdGFuY2UuXG4gICAqL1xuICBkdjogRGF0YXZpZXdJbmxpbmVBcGk7XG5cbiAgLyoqXG4gICAqIEEgaGVhZGVyIHRleHQgb2YgdGhlIGNhbGxvdXQsIGRlZmF1bHQgaXMgYW4gZW1wdHkgc3RyaW5nLlxuICAgKi9cbiAgaGVhZGVyPzogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBBIGNhbGxvdXQgbW9kZSwgZGVmYXVsdCBpcyBgQ2FsbG91dE1vZGUuRm9sZGFibGVDb2xsYXBzZWRgLlxuICAgKi9cbiAgbW9kZT86IENhbGxvdXRNb2RlO1xuXG4gIC8qKlxuICAgKiBBIHR5cGUgb2YgdGhlIGNhbGxvdXQsIGRlZmF1bHQgaXMgYFwiTk9URVwiYC5cbiAgICovXG4gIHR5cGU/OiBzdHJpbmc7XG59XG5cbi8qKlxuICogUmVuZGVycyBhIGNhbGxvdXQgYmxvY2sgaW4gRGF0YXZpZXcuXG4gKlxuICogQHBhcmFtIG9wdGlvbnMgLSBUaGUgb3B0aW9ucyBmb3IgcmVuZGVyaW5nIHRoZSBjYWxsb3V0LlxuICovXG5leHBvcnQgZnVuY3Rpb24gcmVuZGVyQ2FsbG91dChvcHRpb25zOiBSZW5kZXJDYWxsb3V0T3B0aW9ucyk6IHZvaWQge1xuICBjb25zdCB7XG4gICAgY29udGVudFByb3ZpZGVyID0gJycsXG4gICAgZHYsXG4gICAgaGVhZGVyID0gJycsXG4gICAgbW9kZSA9IENhbGxvdXRNb2RlLkZvbGRhYmxlQ29sbGFwc2VkLFxuICAgIHR5cGUgPSAnTk9URSdcbiAgfSA9IG9wdGlvbnM7XG4gIGNvbnN0IG1vZGlmaWVyID0gZ2V0TW9kaWZpZXIobW9kZSk7XG4gIGNvbnN0IGNhbGxvdXQgPSBkdi5wYXJhZ3JhcGgoYD4gWyEke3R5cGV9XSR7bW9kaWZpZXJ9ICR7aGVhZGVyfVxcbj5cXG4+IDxkaXYgY2xhc3M9XCJjb250ZW50XCI+PC9kaXY+YCk7XG4gIGNvbnN0IGNvbnRlbnREaXYgPSBjYWxsb3V0LnF1ZXJ5U2VsZWN0b3I8SFRNTERpdkVsZW1lbnQ+KCcuY29udGVudCcpID8/IHRocm93RXhwcmVzc2lvbihuZXcgRXJyb3IoJ0NvbnRlbnQgZGl2IG5vdCBmb3VuZCcpKTtcbiAgZHYucGFyYWdyYXBoKCdMb2FkaW5nLi4uIFx1MjNGMycsIHsgY29udGFpbmVyOiBjb250ZW50RGl2IH0pO1xuXG4gIGNvbnN0IG9ic2VydmVyID0gbmV3IEludGVyc2VjdGlvbk9ic2VydmVyKChlbnRyaWVzOiBJbnRlcnNlY3Rpb25PYnNlcnZlckVudHJ5W10pID0+IHtcbiAgICBmb3IgKGNvbnN0IGVudHJ5IG9mIGVudHJpZXMpIHtcbiAgICAgIGlmIChlbnRyeS5pc0ludGVyc2VjdGluZykge1xuICAgICAgICBvYnNlcnZlci51bm9ic2VydmUoZW50cnkudGFyZ2V0KTtcbiAgICAgICAgYWRkVG9RdWV1ZShub3JtYWxpemVPcHRpb25hbFByb3BlcnRpZXM8QWRkVG9RdWV1ZU9wdGlvbnM+KHtcbiAgICAgICAgICBhYm9ydFNpZ25hbDogb3B0aW9ucy5hYm9ydFNpZ25hbCxcbiAgICAgICAgICBhcHA6IGR2LmFwcCxcbiAgICAgICAgICBvcGVyYXRpb25GbjogbG9hZENvbnRlbnQsXG4gICAgICAgICAgb3BlcmF0aW9uTmFtZTogJ0xvYWQgY29udGVudCBmb3IgY2FsbG91dCdcbiAgICAgICAgfSkpO1xuICAgICAgfVxuICAgIH1cbiAgfSk7XG4gIG9ic2VydmVyLm9ic2VydmUoY29udGVudERpdik7XG5cbiAgYXN5bmMgZnVuY3Rpb24gbG9hZENvbnRlbnQoYWJvcnRTaWduYWw6IEFib3J0U2lnbmFsKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgYWJvcnRTaWduYWwudGhyb3dJZkFib3J0ZWQoKTtcbiAgICBsZXQgY29udGVudDogTWF5YmVSZXR1cm48Tm9kZSB8IHN0cmluZyB8IHVuZGVmaW5lZD47XG5cbiAgICBjb25zdCBwYXJhZ3JhcGggPSBhd2FpdCBnZXRSZW5kZXJlZENvbnRhaW5lcihkdiwgYXN5bmMgKCkgPT4ge1xuICAgICAgY29udGVudCA9IGF3YWl0IHJlc29sdmVWYWx1ZShjb250ZW50UHJvdmlkZXIsIGFib3J0U2lnbmFsKTtcbiAgICAgIGFib3J0U2lnbmFsLnRocm93SWZBYm9ydGVkKCk7XG4gICAgfSk7XG4gICAgYWJvcnRTaWduYWwudGhyb3dJZkFib3J0ZWQoKTtcblxuICAgIGNvbnRlbnQgPz89IHBhcmFncmFwaDtcblxuICAgIGNvbnRlbnREaXYuZW1wdHkoKTtcbiAgICBkdi5wYXJhZ3JhcGgoY29udGVudCwgeyBjb250YWluZXI6IGNvbnRlbnREaXYgfSk7XG4gIH1cbn1cblxuLyoqXG4gKiBXcmFwcyB0aGUgcHJvdmlkZWQgY29udGVudCBpbiBibG9ja3F1b3RlIHN5bnRheCBmb3IgYSBjYWxsb3V0LlxuICpcbiAqIEBwYXJhbSBjb250ZW50IC0gVGhlIGNvbnRlbnQgdG8gd3JhcC5cbiAqIEByZXR1cm5zIFRoZSBjb250ZW50IHdyYXBwZWQgaW4gYmxvY2txdW90ZSBzeW50YXguXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiB3cmFwRm9yQ2FsbG91dChjb250ZW50OiBzdHJpbmcpOiBzdHJpbmcge1xuICBjb25zdCBsaW5lcyA9IGNvbnRlbnQuc3BsaXQoJ1xcbicpO1xuICBjb25zdCBwcmVmaXhlZExpbmVzID0gbGluZXMubWFwKChsaW5lKSA9PiBgPiAke2xpbmV9YCk7XG4gIHJldHVybiBwcmVmaXhlZExpbmVzLmpvaW4oJ1xcbicpO1xufVxuXG4vKipcbiAqIFJldHVybnMgdGhlIG1vZGlmaWVyIHN0cmluZyBiYXNlZCBvbiB0aGUgY2FsbG91dCBtb2RlLlxuICpcbiAqIEBwYXJhbSBtb2RlIC0gVGhlIG1vZGUgb2YgdGhlIGNhbGxvdXQuXG4gKiBAcmV0dXJucyBUaGUgY29ycmVzcG9uZGluZyBtb2RpZmllciBzdHJpbmcuXG4gKi9cbmZ1bmN0aW9uIGdldE1vZGlmaWVyKG1vZGU6IENhbGxvdXRNb2RlKTogc3RyaW5nIHtcbiAgc3dpdGNoIChtb2RlKSB7XG4gICAgY2FzZSBDYWxsb3V0TW9kZS5Gb2xkYWJsZUNvbGxhcHNlZDpcbiAgICAgIHJldHVybiAnLSc7XG4gICAgY2FzZSBDYWxsb3V0TW9kZS5Gb2xkYWJsZUV4cGFuZGVkOlxuICAgICAgcmV0dXJuICcrJztcbiAgICBkZWZhdWx0OlxuICAgICAgcmV0dXJuICcnO1xuICB9XG59XG4iXSwKICAibWFwcGluZ3MiOiAiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFXQSxtQkFBZ0M7QUFDaEMseUJBQTRDO0FBQzVDLDJCQUE2QjtBQUM3QixzQkFBcUM7QUFDckMsbUJBQTJCO0FBS3BCLElBQUssY0FBTCxrQkFBS0EsaUJBQUw7QUFFTCxFQUFBQSwwQkFBQTtBQUdBLEVBQUFBLDBCQUFBO0FBR0EsRUFBQUEsMEJBQUE7QUFSVSxTQUFBQTtBQUFBLEdBQUE7QUFtREwsU0FBUyxjQUFjLFNBQXFDO0FBQ2pFLFFBQU07QUFBQSxJQUNKLGtCQUFrQjtBQUFBLElBQ2xCO0FBQUEsSUFDQSxTQUFTO0FBQUEsSUFDVCxPQUFPO0FBQUEsSUFDUCxPQUFPO0FBQUEsRUFDVCxJQUFJO0FBQ0osUUFBTSxXQUFXLFlBQVksSUFBSTtBQUNqQyxRQUFNLFVBQVUsR0FBRyxVQUFVLE9BQU8sSUFBSSxJQUFJLFFBQVEsSUFBSSxNQUFNO0FBQUE7QUFBQSw4QkFBb0M7QUFDbEcsUUFBTSxhQUFhLFFBQVEsY0FBOEIsVUFBVSxTQUFLLDhCQUFnQixJQUFJLE1BQU0sdUJBQXVCLENBQUM7QUFDMUgsS0FBRyxVQUFVLHFCQUFnQixFQUFFLFdBQVcsV0FBVyxDQUFDO0FBRXRELFFBQU0sV0FBVyxJQUFJLHFCQUFxQixDQUFDLFlBQXlDO0FBQ2xGLGVBQVcsU0FBUyxTQUFTO0FBQzNCLFVBQUksTUFBTSxnQkFBZ0I7QUFDeEIsaUJBQVMsVUFBVSxNQUFNLE1BQU07QUFDL0IseUNBQVcsZ0RBQStDO0FBQUEsVUFDeEQsYUFBYSxRQUFRO0FBQUEsVUFDckIsS0FBSyxHQUFHO0FBQUEsVUFDUixhQUFhO0FBQUEsVUFDYixlQUFlO0FBQUEsUUFDakIsQ0FBQyxDQUFDO0FBQUEsTUFDSjtBQUFBLElBQ0Y7QUFBQSxFQUNGLENBQUM7QUFDRCxXQUFTLFFBQVEsVUFBVTtBQUUzQixpQkFBZSxZQUFZLGFBQXlDO0FBQ2xFLGdCQUFZLGVBQWU7QUFDM0IsUUFBSTtBQUVKLFVBQU0sWUFBWSxVQUFNLHNDQUFxQixJQUFJLFlBQVk7QUFDM0QsZ0JBQVUsVUFBTSxtQ0FBYSxpQkFBaUIsV0FBVztBQUN6RCxrQkFBWSxlQUFlO0FBQUEsSUFDN0IsQ0FBQztBQUNELGdCQUFZLGVBQWU7QUFFM0IsZ0JBQVk7QUFFWixlQUFXLE1BQU07QUFDakIsT0FBRyxVQUFVLFNBQVMsRUFBRSxXQUFXLFdBQVcsQ0FBQztBQUFBLEVBQ2pEO0FBQ0Y7QUFRTyxTQUFTLGVBQWUsU0FBeUI7QUFDdEQsUUFBTSxRQUFRLFFBQVEsTUFBTSxJQUFJO0FBQ2hDLFFBQU0sZ0JBQWdCLE1BQU0sSUFBSSxDQUFDLFNBQVMsS0FBSyxJQUFJLEVBQUU7QUFDckQsU0FBTyxjQUFjLEtBQUssSUFBSTtBQUNoQztBQVFBLFNBQVMsWUFBWSxNQUEyQjtBQUM5QyxVQUFRLE1BQU07QUFBQSxJQUNaLEtBQUs7QUFDSCxhQUFPO0FBQUEsSUFDVCxLQUFLO0FBQ0gsYUFBTztBQUFBLElBQ1Q7QUFDRSxhQUFPO0FBQUEsRUFDWDtBQUNGOyIsCiAgIm5hbWVzIjogWyJDYWxsb3V0TW9kZSJdCn0K
|
|
@@ -137,9 +137,9 @@ __export(MetadataCache_exports, {
|
|
|
137
137
|
});
|
|
138
138
|
module.exports = __toCommonJS(MetadataCache_exports);
|
|
139
139
|
var import_implementations = require('obsidian-typings/implementations');
|
|
140
|
-
var import_Async = require('../Async.cjs');
|
|
141
140
|
var import_ObjectUtils = require('../ObjectUtils.cjs');
|
|
142
141
|
var import_App = require('./App.cjs');
|
|
142
|
+
var import_AsyncWithNotice = require('./AsyncWithNotice.cjs');
|
|
143
143
|
var import_FileSystem = require('./FileSystem.cjs');
|
|
144
144
|
var import_Frontmatter = require('./Frontmatter.cjs');
|
|
145
145
|
var import_FrontmatterLinkCacheWithOffsets = require('./FrontmatterLinkCacheWithOffsets.cjs');
|
|
@@ -192,51 +192,55 @@ async function getBacklinksForFileSafe(app, pathOrFile, retryOptions = {}) {
|
|
|
192
192
|
return safeOverload(pathOrFile);
|
|
193
193
|
}
|
|
194
194
|
let backlinks = new import_implementations.CustomArrayDictImpl();
|
|
195
|
-
await (0,
|
|
196
|
-
abortSignal
|
|
197
|
-
const file = (0, import_FileSystem.getFile)(app, pathOrFile);
|
|
198
|
-
await ensureMetadataCacheReady(app);
|
|
199
|
-
abortSignal.throwIfAborted();
|
|
200
|
-
backlinks = getBacklinksForFileOrPath(app, file);
|
|
201
|
-
for (const notePath of backlinks.keys()) {
|
|
195
|
+
await (0, import_AsyncWithNotice.retryWithTimeoutNotice)({
|
|
196
|
+
async operationFn(abortSignal) {
|
|
202
197
|
abortSignal.throwIfAborted();
|
|
203
|
-
const
|
|
204
|
-
|
|
205
|
-
return false;
|
|
206
|
-
}
|
|
207
|
-
await (0, import_Vault.saveNote)(app, note);
|
|
208
|
-
abortSignal.throwIfAborted();
|
|
209
|
-
const content = await (0, import_Vault.readSafe)(app, note);
|
|
198
|
+
const file = (0, import_FileSystem.getFile)(app, pathOrFile);
|
|
199
|
+
await ensureMetadataCacheReady(app);
|
|
210
200
|
abortSignal.throwIfAborted();
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
const propertyValue = (0, import_ObjectUtils.getNestedPropertyValue)(frontmatter, link.key);
|
|
225
|
-
if (typeof propertyValue !== "string") {
|
|
226
|
-
return false;
|
|
227
|
-
}
|
|
228
|
-
const linkWithOffsets = (0, import_FrontmatterLinkCacheWithOffsets.toFrontmatterLinkCacheWithOffsets)(link);
|
|
229
|
-
actualLink = propertyValue.slice(linkWithOffsets.startOffset, linkWithOffsets.endOffset);
|
|
230
|
-
} else {
|
|
231
|
-
return true;
|
|
201
|
+
backlinks = getBacklinksForFileOrPath(app, file);
|
|
202
|
+
for (const notePath of backlinks.keys()) {
|
|
203
|
+
abortSignal.throwIfAborted();
|
|
204
|
+
const note = (0, import_FileSystem.getFileOrNull)(app, notePath);
|
|
205
|
+
if (!note) {
|
|
206
|
+
return false;
|
|
207
|
+
}
|
|
208
|
+
await (0, import_Vault.saveNote)(app, note);
|
|
209
|
+
abortSignal.throwIfAborted();
|
|
210
|
+
const content = await (0, import_Vault.readSafe)(app, note);
|
|
211
|
+
abortSignal.throwIfAborted();
|
|
212
|
+
if (!content) {
|
|
213
|
+
return false;
|
|
232
214
|
}
|
|
233
|
-
|
|
215
|
+
const frontmatter = (0, import_Frontmatter.parseFrontmatter)(content);
|
|
216
|
+
const links = backlinks.get(notePath);
|
|
217
|
+
if (!links) {
|
|
234
218
|
return false;
|
|
235
219
|
}
|
|
220
|
+
for (const link of links) {
|
|
221
|
+
let actualLink;
|
|
222
|
+
if ((0, import_implementations.isReferenceCache)(link)) {
|
|
223
|
+
actualLink = content.slice(link.position.start.offset, link.position.end.offset);
|
|
224
|
+
} else if ((0, import_implementations.isFrontmatterLinkCache)(link)) {
|
|
225
|
+
const propertyValue = (0, import_ObjectUtils.getNestedPropertyValue)(frontmatter, link.key);
|
|
226
|
+
if (typeof propertyValue !== "string") {
|
|
227
|
+
return false;
|
|
228
|
+
}
|
|
229
|
+
const linkWithOffsets = (0, import_FrontmatterLinkCacheWithOffsets.toFrontmatterLinkCacheWithOffsets)(link);
|
|
230
|
+
actualLink = propertyValue.slice(linkWithOffsets.startOffset, linkWithOffsets.endOffset);
|
|
231
|
+
} else {
|
|
232
|
+
return true;
|
|
233
|
+
}
|
|
234
|
+
if (actualLink !== link.original) {
|
|
235
|
+
return false;
|
|
236
|
+
}
|
|
237
|
+
}
|
|
236
238
|
}
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
239
|
+
return true;
|
|
240
|
+
},
|
|
241
|
+
operationName: `Get backlinks for ${(0, import_FileSystem.getPath)(app, pathOrFile)}`,
|
|
242
|
+
retryOptions
|
|
243
|
+
});
|
|
240
244
|
return backlinks;
|
|
241
245
|
}
|
|
242
246
|
async function getCacheSafe(app, fileOrPath) {
|
|
@@ -360,4 +364,4 @@ function getRegisteredFilesCounts(app) {
|
|
|
360
364
|
unregisterFileCacheForNonExistingFile,
|
|
361
365
|
unregisterFiles
|
|
362
366
|
});
|
|
363
|
-
//# 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": []
}

|
|
367
|
+
//# 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": "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",
  "names": []
}

|
|
@@ -131,24 +131,30 @@ var import_Async = require('../Async.cjs');
|
|
|
131
131
|
var import_Error = require('../Error.cjs');
|
|
132
132
|
var import_Function = require('../Function.cjs');
|
|
133
133
|
var import_App = require('./App.cjs');
|
|
134
|
+
var import_AsyncWithNotice = require('./AsyncWithNotice.cjs');
|
|
134
135
|
var import_Logger = require('./Logger.cjs');
|
|
135
|
-
function addToQueue(
|
|
136
|
-
stackTrace
|
|
137
|
-
(0, import_Async.invokeAsyncSafely)(() => addToQueueAndWait(
|
|
136
|
+
function addToQueue(options) {
|
|
137
|
+
const stackTrace = options.stackTrace ?? (0, import_Error.getStackTrace)(1);
|
|
138
|
+
(0, import_Async.invokeAsyncSafely)(() => addToQueueAndWait(options), stackTrace);
|
|
138
139
|
}
|
|
139
|
-
async function addToQueueAndWait(
|
|
140
|
-
abortSignal
|
|
140
|
+
async function addToQueueAndWait(options) {
|
|
141
|
+
const abortSignal = options.abortSignal ?? (0, import_AbortController.abortSignalNever)();
|
|
141
142
|
abortSignal.throwIfAborted();
|
|
142
143
|
const DEFAULT_TIMEOUT_IN_MILLISECONDS = 6e4;
|
|
143
|
-
timeoutInMilliseconds
|
|
144
|
-
stackTrace
|
|
145
|
-
const
|
|
146
|
-
queue
|
|
147
|
-
queue.
|
|
144
|
+
const timeoutInMilliseconds = options.timeoutInMilliseconds ?? DEFAULT_TIMEOUT_IN_MILLISECONDS;
|
|
145
|
+
const stackTrace = options.stackTrace ?? (0, import_Error.getStackTrace)(1);
|
|
146
|
+
const operationName = options.operationName ?? "";
|
|
147
|
+
const queue = getQueue(options.app).value;
|
|
148
|
+
queue.items.push({ abortSignal, operationFn: options.operationFn, operationName, stackTrace, timeoutInMilliseconds });
|
|
149
|
+
queue.promise = queue.promise.then(() => processNextQueueItem(options.app));
|
|
148
150
|
await queue.promise;
|
|
149
151
|
}
|
|
150
152
|
async function flushQueue(app) {
|
|
151
|
-
await addToQueueAndWait(
|
|
153
|
+
await addToQueueAndWait({
|
|
154
|
+
app,
|
|
155
|
+
operationFn: import_Function.noop,
|
|
156
|
+
operationName: "Flush queue"
|
|
157
|
+
});
|
|
152
158
|
}
|
|
153
159
|
function getQueue(app) {
|
|
154
160
|
return (0, import_App.getObsidianDevUtilsState)(app, "queue", { items: [], promise: Promise.resolve() });
|
|
@@ -159,11 +165,22 @@ async function processNextQueueItem(app) {
|
|
|
159
165
|
if (!item) {
|
|
160
166
|
return;
|
|
161
167
|
}
|
|
162
|
-
await (0, import_Async.addErrorHandler)(
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
168
|
+
await (0, import_Async.addErrorHandler)(
|
|
169
|
+
() => (0, import_AsyncWithNotice.runWithTimeoutNotice)({
|
|
170
|
+
context: { queuedFn: item.operationFn },
|
|
171
|
+
async operationFn(abortSignal) {
|
|
172
|
+
await (0, import_Logger.invokeAsyncAndLog)(
|
|
173
|
+
item.operationName || processNextQueueItem.name,
|
|
174
|
+
item.operationFn,
|
|
175
|
+
(0, import_AbortController.abortSignalAny)(abortSignal, item.abortSignal),
|
|
176
|
+
item.stackTrace
|
|
177
|
+
);
|
|
178
|
+
},
|
|
179
|
+
operationName: item.operationName,
|
|
180
|
+
stackTrace: item.stackTrace,
|
|
181
|
+
timeoutInMilliseconds: item.timeoutInMilliseconds
|
|
182
|
+
})
|
|
183
|
+
);
|
|
167
184
|
queue.items.shift();
|
|
168
185
|
}
|
|
169
186
|
// Annotate the CommonJS export names for ESM import in node:
|
|
@@ -172,4 +189,4 @@ async function processNextQueueItem(app) {
|
|
|
172
189
|
addToQueueAndWait,
|
|
173
190
|
flushQueue
|
|
174
191
|
});
|
|
175
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
192
|
+
//# 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": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAWA,6BAGO;AACP,mBAGO;AACP,mBAA8B;AAC9B,sBAAqB;AACrB,iBAAyC;AACzC,6BAAqC;AACrC,oBAAkC;AA0F3B,SAAS,WAAW,SAAkC;AAC3D,QAAM,aAAa,QAAQ,kBAAc,4BAAc,CAAC;AACxD,sCAAkB,MAAM,kBAAkB,OAAO,GAAG,UAAU;AAChE;AAOA,eAAsB,kBAAkB,SAAkD;AACxF,QAAM,cAAc,QAAQ,mBAAe,yCAAiB;AAC5D,cAAY,eAAe;AAE3B,QAAM,kCAAkC;AACxC,QAAM,wBAAwB,QAAQ,yBAAyB;AAC/D,QAAM,aAAa,QAAQ,kBAAc,4BAAc,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,aAAO,qCAAyB,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,YAAM;AAAA,IAAgB,UACpB,6CAAqB;AAAA,MACnB,SAAS,EAAE,UAAU,KAAK,YAAY;AAAA,MACtC,MAAM,YAAY,aAAyC;AACzD,kBAAM;AAAA,UACJ,KAAK,iBAAiB,qBAAqB;AAAA,UAC3C,KAAK;AAAA,cACL,uCAAe,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": []
}

|