obsidian-dev-utils 20.1.0 → 20.2.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 +5 -0
- package/dist/lib/cjs/Library.cjs +1 -1
- package/dist/lib/cjs/obsidian/FileChange.cjs +86 -84
- package/dist/lib/cjs/obsidian/FileChange.d.cts +9 -0
- package/dist/lib/cjs/obsidian/Link.cjs +67 -29
- package/dist/lib/cjs/obsidian/Link.d.cts +50 -0
- package/dist/lib/esm/Library.mjs +1 -1
- package/dist/lib/esm/obsidian/FileChange.d.mts +9 -0
- package/dist/lib/esm/obsidian/FileChange.mjs +85 -84
- package/dist/lib/esm/obsidian/Link.d.mts +50 -0
- package/dist/lib/esm/obsidian/Link.mjs +67 -29
- package/package.json +4 -4
@@ -21,6 +21,89 @@ import {
|
|
21
21
|
} from "./Frontmatter.mjs";
|
22
22
|
import { referenceToFileChange } from "./Reference.mjs";
|
23
23
|
import { process } from "./Vault.mjs";
|
24
|
+
async function applyContentChanges(content, path, changesProvider) {
|
25
|
+
let changes = await resolveValue(changesProvider);
|
26
|
+
const frontmatter = parseFrontmatter(content);
|
27
|
+
for (const change of changes) {
|
28
|
+
if (isContentChange(change)) {
|
29
|
+
const actualContent = content.slice(change.startIndex, change.endIndex);
|
30
|
+
if (actualContent !== change.oldContent) {
|
31
|
+
console.warn("Content mismatch", {
|
32
|
+
actualContent,
|
33
|
+
endIndex: change.endIndex,
|
34
|
+
expectedContent: change.oldContent,
|
35
|
+
path,
|
36
|
+
startIndex: change.startIndex
|
37
|
+
});
|
38
|
+
return null;
|
39
|
+
}
|
40
|
+
} else if (isFrontmatterChange(change)) {
|
41
|
+
const actualContent = getNestedPropertyValue(frontmatter, change.frontmatterKey);
|
42
|
+
if (actualContent !== change.oldContent) {
|
43
|
+
console.warn("Content mismatch", {
|
44
|
+
actualContent,
|
45
|
+
expectedContent: change.oldContent,
|
46
|
+
frontmatterKey: change.frontmatterKey,
|
47
|
+
path
|
48
|
+
});
|
49
|
+
return null;
|
50
|
+
}
|
51
|
+
}
|
52
|
+
}
|
53
|
+
changes.sort((a, b) => {
|
54
|
+
if (isContentChange(a) && isContentChange(b)) {
|
55
|
+
return a.startIndex - b.startIndex;
|
56
|
+
}
|
57
|
+
if (isFrontmatterChange(a) && isFrontmatterChange(b)) {
|
58
|
+
return a.frontmatterKey.localeCompare(b.frontmatterKey);
|
59
|
+
}
|
60
|
+
return isContentChange(a) ? -1 : 1;
|
61
|
+
});
|
62
|
+
changes = changes.filter((change, index) => {
|
63
|
+
if (change.oldContent === change.newContent) {
|
64
|
+
return false;
|
65
|
+
}
|
66
|
+
if (index === 0) {
|
67
|
+
return true;
|
68
|
+
}
|
69
|
+
return !deepEqual(change, changes[index - 1]);
|
70
|
+
});
|
71
|
+
for (let i = 1; i < changes.length; i++) {
|
72
|
+
const change = changes[i];
|
73
|
+
if (!change) {
|
74
|
+
continue;
|
75
|
+
}
|
76
|
+
const previousChange = changes[i - 1];
|
77
|
+
if (!previousChange) {
|
78
|
+
continue;
|
79
|
+
}
|
80
|
+
if (isContentChange(previousChange) && isContentChange(change) && previousChange.endIndex && change.startIndex && previousChange.endIndex > change.startIndex) {
|
81
|
+
console.warn("Overlapping changes", {
|
82
|
+
change,
|
83
|
+
previousChange
|
84
|
+
});
|
85
|
+
return null;
|
86
|
+
}
|
87
|
+
}
|
88
|
+
let newContent = "";
|
89
|
+
let lastIndex = 0;
|
90
|
+
let frontmatterChanged = false;
|
91
|
+
for (const change of changes) {
|
92
|
+
if (isContentChange(change)) {
|
93
|
+
newContent += content.slice(lastIndex, change.startIndex);
|
94
|
+
newContent += change.newContent;
|
95
|
+
lastIndex = change.endIndex;
|
96
|
+
} else if (isFrontmatterChange(change)) {
|
97
|
+
setNestedPropertyValue(frontmatter, change.frontmatterKey, change.newContent);
|
98
|
+
frontmatterChanged = true;
|
99
|
+
}
|
100
|
+
}
|
101
|
+
newContent += content.slice(lastIndex);
|
102
|
+
if (frontmatterChanged) {
|
103
|
+
newContent = setFrontmatter(newContent, frontmatter);
|
104
|
+
}
|
105
|
+
return newContent;
|
106
|
+
}
|
24
107
|
async function applyFileChanges(app, pathOrFile, changesProvider, processOptions = {}) {
|
25
108
|
await process(app, pathOrFile, async (content) => {
|
26
109
|
if (isCanvasFile(app, pathOrFile)) {
|
@@ -106,89 +189,6 @@ async function applyCanvasChanges(content, path, changesProvider) {
|
|
106
189
|
}
|
107
190
|
return JSON.stringify(canvasData, null, " ");
|
108
191
|
}
|
109
|
-
async function applyContentChanges(content, path, changesProvider) {
|
110
|
-
let changes = await resolveValue(changesProvider);
|
111
|
-
const frontmatter = parseFrontmatter(content);
|
112
|
-
for (const change of changes) {
|
113
|
-
if (isContentChange(change)) {
|
114
|
-
const actualContent = content.slice(change.startIndex, change.endIndex);
|
115
|
-
if (actualContent !== change.oldContent) {
|
116
|
-
console.warn("Content mismatch", {
|
117
|
-
actualContent,
|
118
|
-
endIndex: change.endIndex,
|
119
|
-
expectedContent: change.oldContent,
|
120
|
-
path,
|
121
|
-
startIndex: change.startIndex
|
122
|
-
});
|
123
|
-
return null;
|
124
|
-
}
|
125
|
-
} else if (isFrontmatterChange(change)) {
|
126
|
-
const actualContent = getNestedPropertyValue(frontmatter, change.frontmatterKey);
|
127
|
-
if (actualContent !== change.oldContent) {
|
128
|
-
console.warn("Content mismatch", {
|
129
|
-
actualContent,
|
130
|
-
expectedContent: change.oldContent,
|
131
|
-
frontmatterKey: change.frontmatterKey,
|
132
|
-
path
|
133
|
-
});
|
134
|
-
return null;
|
135
|
-
}
|
136
|
-
}
|
137
|
-
}
|
138
|
-
changes.sort((a, b) => {
|
139
|
-
if (isContentChange(a) && isContentChange(b)) {
|
140
|
-
return a.startIndex - b.startIndex;
|
141
|
-
}
|
142
|
-
if (isFrontmatterChange(a) && isFrontmatterChange(b)) {
|
143
|
-
return a.frontmatterKey.localeCompare(b.frontmatterKey);
|
144
|
-
}
|
145
|
-
return isContentChange(a) ? -1 : 1;
|
146
|
-
});
|
147
|
-
changes = changes.filter((change, index) => {
|
148
|
-
if (change.oldContent === change.newContent) {
|
149
|
-
return false;
|
150
|
-
}
|
151
|
-
if (index === 0) {
|
152
|
-
return true;
|
153
|
-
}
|
154
|
-
return !deepEqual(change, changes[index - 1]);
|
155
|
-
});
|
156
|
-
for (let i = 1; i < changes.length; i++) {
|
157
|
-
const change = changes[i];
|
158
|
-
if (!change) {
|
159
|
-
continue;
|
160
|
-
}
|
161
|
-
const previousChange = changes[i - 1];
|
162
|
-
if (!previousChange) {
|
163
|
-
continue;
|
164
|
-
}
|
165
|
-
if (isContentChange(previousChange) && isContentChange(change) && previousChange.endIndex && change.startIndex && previousChange.endIndex > change.startIndex) {
|
166
|
-
console.warn("Overlapping changes", {
|
167
|
-
change,
|
168
|
-
previousChange
|
169
|
-
});
|
170
|
-
return null;
|
171
|
-
}
|
172
|
-
}
|
173
|
-
let newContent = "";
|
174
|
-
let lastIndex = 0;
|
175
|
-
let frontmatterChanged = false;
|
176
|
-
for (const change of changes) {
|
177
|
-
if (isContentChange(change)) {
|
178
|
-
newContent += content.slice(lastIndex, change.startIndex);
|
179
|
-
newContent += change.newContent;
|
180
|
-
lastIndex = change.endIndex;
|
181
|
-
} else if (isFrontmatterChange(change)) {
|
182
|
-
setNestedPropertyValue(frontmatter, change.frontmatterKey, change.newContent);
|
183
|
-
frontmatterChanged = true;
|
184
|
-
}
|
185
|
-
}
|
186
|
-
newContent += content.slice(lastIndex);
|
187
|
-
if (frontmatterChanged) {
|
188
|
-
newContent = setFrontmatter(newContent, frontmatter);
|
189
|
-
}
|
190
|
-
return newContent;
|
191
|
-
}
|
192
192
|
function parseJsonSafe(content) {
|
193
193
|
let parsed;
|
194
194
|
try {
|
@@ -202,6 +202,7 @@ function parseJsonSafe(content) {
|
|
202
202
|
return parsed;
|
203
203
|
}
|
204
204
|
export {
|
205
|
+
applyContentChanges,
|
205
206
|
applyFileChanges,
|
206
207
|
isCanvasChange,
|
207
208
|
isCanvasFileNodeChange,
|
@@ -209,4 +210,4 @@ export {
|
|
209
210
|
isContentChange,
|
210
211
|
isFrontmatterChange
|
211
212
|
};
|
212
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/FileChange.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation FileChange\n * Contains utility types and functions for handling file changes in Obsidian.\n */\n\nimport type {\n  App,\n  Reference\n} from 'obsidian';\nimport type { CanvasData } from 'obsidian/Canvas.d.ts';\n\nimport type { ValueProvider } from '../ValueProvider.ts';\nimport type { PathOrFile } from './FileSystem.ts';\nimport type { ProcessOptions } from './Vault.ts';\n\nimport {\n  deepEqual,\n  getNestedPropertyValue,\n  setNestedPropertyValue\n} from '../Object.ts';\nimport { resolveValue } from '../ValueProvider.ts';\nimport {\n  getPath,\n  isCanvasFile\n} from './FileSystem.ts';\nimport {\n  parseFrontmatter,\n  setFrontmatter\n} from './Frontmatter.ts';\nimport { referenceToFileChange } from './Reference.ts';\nimport { process } from './Vault.ts';\n\n/**\n * Represents a canvas change in the Vault.\n */\nexport interface CanvasChange extends FileChange {\n  /**\n   * Whether the change is a canvas change.\n   */\n  isCanvas: true;\n\n  /**\n   * The index of the node in the canvas.\n   */\n  nodeIndex: number;\n\n  /**\n   * The type of link.\n   */\n  type: 'file' | 'text';\n}\n\n/**\n * Represents a change in a file node in a canvas.\n */\nexport interface CanvasFileNodeChange extends CanvasChange {\n  /**\n   * The type of link.\n   */\n  type: 'file';\n}\n\n/**\n * Represents a change in a text node in a canvas.\n */\nexport interface CanvasTextNodeChange extends CanvasChange {\n  /**\n   * The original reference.\n   */\n  originalReference: Reference;\n\n  /**\n   * The type of link.\n   */\n  type: 'text';\n}\n\n/**\n * Represents a content body change in the Vault.\n */\nexport interface ContentChange extends FileChange {\n  /**\n   * The end index of the change in the file content.\n   */\n  endIndex: number;\n\n  /**\n   * The start index of the change in the file content.\n   */\n  startIndex: number;\n}\n\n/**\n * Represents a file change in the Vault.\n */\nexport interface FileChange {\n  /**\n   * The new content to replace the old content.\n   */\n  newContent: string;\n\n  /**\n   * The old content that will be replaced.\n   */\n  oldContent: string;\n}\n\n/**\n * Represents a frontmatter change in the Vault.\n */\nexport interface FrontmatterChange extends FileChange {\n  /**\n   * The key in the frontmatter to use for the link.\n   */\n  frontmatterKey: string;\n}\n\n/**\n * Applies a series of file changes to the specified file or path within the application.\n *\n * @param app - The application instance where the file changes will be applied.\n * @param pathOrFile - The path or file to which the changes should be applied.\n * @param changesProvider - A provider that returns an array of file changes to apply.\n * @param processOptions - Optional options for processing/retrying the operation.\n *\n * @returns A promise that resolves when the file changes have been successfully applied.\n */\nexport async function applyFileChanges(\n  app: App,\n  pathOrFile: PathOrFile,\n  changesProvider: ValueProvider<FileChange[]>,\n  processOptions: ProcessOptions = {}\n): Promise<void> {\n  await process(app, pathOrFile, async (content) => {\n    if (isCanvasFile(app, pathOrFile)) {\n      return applyCanvasChanges(content, getPath(app, pathOrFile), changesProvider);\n    }\n\n    return await applyContentChanges(content, getPath(app, pathOrFile), changesProvider);\n  }, processOptions);\n}\n\n/**\n * Checks if a file change is a canvas change.\n *\n * @param change - The file change to check.\n * @returns Whether the file change is a canvas change.\n */\nexport function isCanvasChange(change: FileChange): change is CanvasChange {\n  return !!(change as Partial<CanvasChange>).isCanvas;\n}\n\n/**\n * Checks if a file change is a canvas file node change.\n *\n * @param change - The file change to check.\n * @returns Whether the file change is a canvas file node change.\n */\nexport function isCanvasFileNodeChange(change: FileChange): change is CanvasFileNodeChange {\n  return isCanvasChange(change) && change.type === 'file';\n}\n\n/**\n * Checks if a file change is a canvas text node change.\n *\n * @param change - The file change to check.\n * @returns Whether the file change is a canvas text node change.\n */\nexport function isCanvasTextNodeChange(change: FileChange): change is CanvasTextNodeChange {\n  return isCanvasChange(change) && change.type === 'text';\n}\n\n/**\n * Checks if a file change is a content change.\n *\n * @param fileChange - The file change to check.\n * @returns A boolean indicating whether the file change is a content change.\n */\nexport function isContentChange(fileChange: FileChange): fileChange is ContentChange {\n  return (fileChange as Partial<ContentChange>).startIndex !== undefined;\n}\n\n/**\n * Checks if a file change is a frontmatter change.\n *\n * @param fileChange - The file change to check.\n * @returns A boolean indicating whether the file change is a frontmatter change.\n */\nexport function isFrontmatterChange(fileChange: FileChange): fileChange is FrontmatterChange {\n  return (fileChange as Partial<FrontmatterChange>).frontmatterKey !== undefined;\n}\n\nasync function applyCanvasChanges(content: string, path: string, changesProvider: ValueProvider<FileChange[]>): Promise<null | string> {\n  const changes = await resolveValue(changesProvider);\n  const canvasData = parseJsonSafe(content) as CanvasData;\n\n  const canvasTextChanges = new Map<number, CanvasTextNodeChange[]>();\n\n  for (const change of changes) {\n    if (!isCanvasChange(change)) {\n      console.warn('Only canvas changes are supported for canvas files', {\n        change,\n        path\n      });\n      return null;\n    }\n\n    const node = canvasData.nodes[change.nodeIndex];\n    if (!node) {\n      console.warn('Node not found', {\n        nodeIndex: change.nodeIndex,\n        path\n      });\n      return null;\n    }\n\n    if (isCanvasFileNodeChange(change)) {\n      if (node.file !== change.oldContent) {\n        console.warn('Content mismatch', {\n          actualContent: node.file as string | undefined,\n          expectedContent: change.oldContent,\n          nodeIndex: change.nodeIndex,\n          path,\n          type: 'file'\n        });\n\n        return null;\n      }\n      node.file = change.newContent;\n    } else if (isCanvasTextNodeChange(change)) {\n      let canvasTextChangesForNode = canvasTextChanges.get(change.nodeIndex);\n      if (!canvasTextChangesForNode) {\n        canvasTextChangesForNode = [];\n        canvasTextChanges.set(change.nodeIndex, canvasTextChangesForNode);\n      }\n\n      canvasTextChangesForNode.push(change);\n    }\n  }\n\n  for (const [nodeIndex, canvasTextChangesForNode] of canvasTextChanges.entries()) {\n    const node = canvasData.nodes[nodeIndex];\n    if (!node) {\n      console.warn('Node not found', {\n        nodeIndex,\n        path\n      });\n\n      return null;\n    }\n\n    if (typeof node.text !== 'string') {\n      console.warn('Node text is not a string', {\n        nodeIndex,\n        path\n      });\n\n      return null;\n    }\n\n    const contentChanges = canvasTextChangesForNode.map((change) => referenceToFileChange(change.originalReference, change.newContent));\n    node.text = await applyContentChanges(node.text, `${path}.node${nodeIndex.toString()}.VIRTUAL_FILE.md`, contentChanges);\n  }\n\n  return JSON.stringify(canvasData, null, '\\t');\n}\n\nasync function applyContentChanges(content: string, path: string, changesProvider: ValueProvider<FileChange[]>): Promise<null | string> {\n  let changes = await resolveValue(changesProvider);\n  const frontmatter = parseFrontmatter(content);\n\n  for (const change of changes) {\n    if (isContentChange(change)) {\n      const actualContent = content.slice(change.startIndex, change.endIndex);\n      if (actualContent !== change.oldContent) {\n        console.warn('Content mismatch', {\n          actualContent,\n          endIndex: change.endIndex,\n          expectedContent: change.oldContent,\n          path,\n          startIndex: change.startIndex\n        });\n\n        return null;\n      }\n    } else if (isFrontmatterChange(change)) {\n      const actualContent = getNestedPropertyValue(frontmatter, change.frontmatterKey);\n      if (actualContent !== change.oldContent) {\n        console.warn('Content mismatch', {\n          actualContent,\n          expectedContent: change.oldContent,\n          frontmatterKey: change.frontmatterKey,\n          path\n        });\n\n        return null;\n      }\n    }\n  }\n\n  changes.sort((a, b) => {\n    if (isContentChange(a) && isContentChange(b)) {\n      return a.startIndex - b.startIndex;\n    }\n\n    if (isFrontmatterChange(a) && isFrontmatterChange(b)) {\n      return a.frontmatterKey.localeCompare(b.frontmatterKey);\n    }\n\n    return isContentChange(a) ? -1 : 1;\n  });\n\n  // BUG: https://forum.obsidian.md/t/bug-duplicated-links-in-metadatacache-inside-footnotes/85551\n  changes = changes.filter((change, index) => {\n    if (change.oldContent === change.newContent) {\n      return false;\n    }\n    if (index === 0) {\n      return true;\n    }\n    return !deepEqual(change, changes[index - 1]);\n  });\n\n  for (let i = 1; i < changes.length; i++) {\n    const change = changes[i];\n    if (!change) {\n      continue;\n    }\n    const previousChange = changes[i - 1];\n    if (!previousChange) {\n      continue;\n    }\n\n    if (\n      isContentChange(previousChange) && isContentChange(change) && previousChange.endIndex && change.startIndex\n      && previousChange.endIndex > change.startIndex\n    ) {\n      console.warn('Overlapping changes', {\n        change,\n        previousChange\n      });\n      return null;\n    }\n  }\n\n  let newContent = '';\n  let lastIndex = 0;\n  let frontmatterChanged = false;\n\n  for (const change of changes) {\n    if (isContentChange(change)) {\n      newContent += content.slice(lastIndex, change.startIndex);\n      newContent += change.newContent;\n      lastIndex = change.endIndex;\n    } else if (isFrontmatterChange(change)) {\n      setNestedPropertyValue(frontmatter, change.frontmatterKey, change.newContent);\n      frontmatterChanged = true;\n    }\n  }\n\n  newContent += content.slice(lastIndex);\n  if (frontmatterChanged) {\n    newContent = setFrontmatter(newContent, frontmatter);\n  }\n\n  return newContent;\n}\n\nfunction parseJsonSafe(content: string): Record<string, unknown> {\n  let parsed: unknown;\n  try {\n    parsed = JSON.parse(content);\n  } catch {\n    parsed = null;\n  }\n\n  if (parsed === null || typeof parsed !== 'object') {\n    parsed = {};\n  }\n\n  return parsed as Record<string, unknown>;\n}\n"],
  "mappings": "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",
  "names": []
}

|
213
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/FileChange.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation FileChange\n * Contains utility types and functions for handling file changes in Obsidian.\n */\n\nimport type {\n  App,\n  Reference\n} from 'obsidian';\nimport type { CanvasData } from 'obsidian/Canvas.d.ts';\n\nimport type { ValueProvider } from '../ValueProvider.ts';\nimport type { PathOrFile } from './FileSystem.ts';\nimport type { ProcessOptions } from './Vault.ts';\n\nimport {\n  deepEqual,\n  getNestedPropertyValue,\n  setNestedPropertyValue\n} from '../Object.ts';\nimport { resolveValue } from '../ValueProvider.ts';\nimport {\n  getPath,\n  isCanvasFile\n} from './FileSystem.ts';\nimport {\n  parseFrontmatter,\n  setFrontmatter\n} from './Frontmatter.ts';\nimport { referenceToFileChange } from './Reference.ts';\nimport { process } from './Vault.ts';\n\n/**\n * Represents a canvas change in the Vault.\n */\nexport interface CanvasChange extends FileChange {\n  /**\n   * Whether the change is a canvas change.\n   */\n  isCanvas: true;\n\n  /**\n   * The index of the node in the canvas.\n   */\n  nodeIndex: number;\n\n  /**\n   * The type of link.\n   */\n  type: 'file' | 'text';\n}\n\n/**\n * Represents a change in a file node in a canvas.\n */\nexport interface CanvasFileNodeChange extends CanvasChange {\n  /**\n   * The type of link.\n   */\n  type: 'file';\n}\n\n/**\n * Represents a change in a text node in a canvas.\n */\nexport interface CanvasTextNodeChange extends CanvasChange {\n  /**\n   * The original reference.\n   */\n  originalReference: Reference;\n\n  /**\n   * The type of link.\n   */\n  type: 'text';\n}\n\n/**\n * Represents a content body change in the Vault.\n */\nexport interface ContentChange extends FileChange {\n  /**\n   * The end index of the change in the file content.\n   */\n  endIndex: number;\n\n  /**\n   * The start index of the change in the file content.\n   */\n  startIndex: number;\n}\n\n/**\n * Represents a file change in the Vault.\n */\nexport interface FileChange {\n  /**\n   * The new content to replace the old content.\n   */\n  newContent: string;\n\n  /**\n   * The old content that will be replaced.\n   */\n  oldContent: string;\n}\n\n/**\n * Represents a frontmatter change in the Vault.\n */\nexport interface FrontmatterChange extends FileChange {\n  /**\n   * The key in the frontmatter to use for the link.\n   */\n  frontmatterKey: string;\n}\n\n/**\n * Applies a series of content changes to the specified content.\n *\n * @param content - The content to which the changes should be applied.\n * @param path - The path to which the changes should be applied.\n * @param changesProvider - A provider that returns an array of content changes to apply.\n * @returns A promise that resolves when the content changes have been successfully applied.\n */\nexport async function applyContentChanges(content: string, path: string, changesProvider: ValueProvider<FileChange[]>): Promise<null | string> {\n  let changes = await resolveValue(changesProvider);\n  const frontmatter = parseFrontmatter(content);\n\n  for (const change of changes) {\n    if (isContentChange(change)) {\n      const actualContent = content.slice(change.startIndex, change.endIndex);\n      if (actualContent !== change.oldContent) {\n        console.warn('Content mismatch', {\n          actualContent,\n          endIndex: change.endIndex,\n          expectedContent: change.oldContent,\n          path,\n          startIndex: change.startIndex\n        });\n\n        return null;\n      }\n    } else if (isFrontmatterChange(change)) {\n      const actualContent = getNestedPropertyValue(frontmatter, change.frontmatterKey);\n      if (actualContent !== change.oldContent) {\n        console.warn('Content mismatch', {\n          actualContent,\n          expectedContent: change.oldContent,\n          frontmatterKey: change.frontmatterKey,\n          path\n        });\n\n        return null;\n      }\n    }\n  }\n\n  changes.sort((a, b) => {\n    if (isContentChange(a) && isContentChange(b)) {\n      return a.startIndex - b.startIndex;\n    }\n\n    if (isFrontmatterChange(a) && isFrontmatterChange(b)) {\n      return a.frontmatterKey.localeCompare(b.frontmatterKey);\n    }\n\n    return isContentChange(a) ? -1 : 1;\n  });\n\n  // BUG: https://forum.obsidian.md/t/bug-duplicated-links-in-metadatacache-inside-footnotes/85551\n  changes = changes.filter((change, index) => {\n    if (change.oldContent === change.newContent) {\n      return false;\n    }\n    if (index === 0) {\n      return true;\n    }\n    return !deepEqual(change, changes[index - 1]);\n  });\n\n  for (let i = 1; i < changes.length; i++) {\n    const change = changes[i];\n    if (!change) {\n      continue;\n    }\n    const previousChange = changes[i - 1];\n    if (!previousChange) {\n      continue;\n    }\n\n    if (\n      isContentChange(previousChange) && isContentChange(change) && previousChange.endIndex && change.startIndex\n      && previousChange.endIndex > change.startIndex\n    ) {\n      console.warn('Overlapping changes', {\n        change,\n        previousChange\n      });\n      return null;\n    }\n  }\n\n  let newContent = '';\n  let lastIndex = 0;\n  let frontmatterChanged = false;\n\n  for (const change of changes) {\n    if (isContentChange(change)) {\n      newContent += content.slice(lastIndex, change.startIndex);\n      newContent += change.newContent;\n      lastIndex = change.endIndex;\n    } else if (isFrontmatterChange(change)) {\n      setNestedPropertyValue(frontmatter, change.frontmatterKey, change.newContent);\n      frontmatterChanged = true;\n    }\n  }\n\n  newContent += content.slice(lastIndex);\n  if (frontmatterChanged) {\n    newContent = setFrontmatter(newContent, frontmatter);\n  }\n\n  return newContent;\n}\n\n/**\n * Applies a series of file changes to the specified file or path within the application.\n *\n * @param app - The application instance where the file changes will be applied.\n * @param pathOrFile - The path or file to which the changes should be applied.\n * @param changesProvider - A provider that returns an array of file changes to apply.\n * @param processOptions - Optional options for processing/retrying the operation.\n *\n * @returns A promise that resolves when the file changes have been successfully applied.\n */\nexport async function applyFileChanges(\n  app: App,\n  pathOrFile: PathOrFile,\n  changesProvider: ValueProvider<FileChange[]>,\n  processOptions: ProcessOptions = {}\n): Promise<void> {\n  await process(app, pathOrFile, async (content) => {\n    if (isCanvasFile(app, pathOrFile)) {\n      return applyCanvasChanges(content, getPath(app, pathOrFile), changesProvider);\n    }\n\n    return await applyContentChanges(content, getPath(app, pathOrFile), changesProvider);\n  }, processOptions);\n}\n\n/**\n * Checks if a file change is a canvas change.\n *\n * @param change - The file change to check.\n * @returns Whether the file change is a canvas change.\n */\nexport function isCanvasChange(change: FileChange): change is CanvasChange {\n  return !!(change as Partial<CanvasChange>).isCanvas;\n}\n\n/**\n * Checks if a file change is a canvas file node change.\n *\n * @param change - The file change to check.\n * @returns Whether the file change is a canvas file node change.\n */\nexport function isCanvasFileNodeChange(change: FileChange): change is CanvasFileNodeChange {\n  return isCanvasChange(change) && change.type === 'file';\n}\n\n/**\n * Checks if a file change is a canvas text node change.\n *\n * @param change - The file change to check.\n * @returns Whether the file change is a canvas text node change.\n */\nexport function isCanvasTextNodeChange(change: FileChange): change is CanvasTextNodeChange {\n  return isCanvasChange(change) && change.type === 'text';\n}\n\n/**\n * Checks if a file change is a content change.\n *\n * @param fileChange - The file change to check.\n * @returns A boolean indicating whether the file change is a content change.\n */\nexport function isContentChange(fileChange: FileChange): fileChange is ContentChange {\n  return (fileChange as Partial<ContentChange>).startIndex !== undefined;\n}\n\n/**\n * Checks if a file change is a frontmatter change.\n *\n * @param fileChange - The file change to check.\n * @returns A boolean indicating whether the file change is a frontmatter change.\n */\nexport function isFrontmatterChange(fileChange: FileChange): fileChange is FrontmatterChange {\n  return (fileChange as Partial<FrontmatterChange>).frontmatterKey !== undefined;\n}\n\nasync function applyCanvasChanges(content: string, path: string, changesProvider: ValueProvider<FileChange[]>): Promise<null | string> {\n  const changes = await resolveValue(changesProvider);\n  const canvasData = parseJsonSafe(content) as CanvasData;\n\n  const canvasTextChanges = new Map<number, CanvasTextNodeChange[]>();\n\n  for (const change of changes) {\n    if (!isCanvasChange(change)) {\n      console.warn('Only canvas changes are supported for canvas files', {\n        change,\n        path\n      });\n      return null;\n    }\n\n    const node = canvasData.nodes[change.nodeIndex];\n    if (!node) {\n      console.warn('Node not found', {\n        nodeIndex: change.nodeIndex,\n        path\n      });\n      return null;\n    }\n\n    if (isCanvasFileNodeChange(change)) {\n      if (node.file !== change.oldContent) {\n        console.warn('Content mismatch', {\n          actualContent: node.file as string | undefined,\n          expectedContent: change.oldContent,\n          nodeIndex: change.nodeIndex,\n          path,\n          type: 'file'\n        });\n\n        return null;\n      }\n      node.file = change.newContent;\n    } else if (isCanvasTextNodeChange(change)) {\n      let canvasTextChangesForNode = canvasTextChanges.get(change.nodeIndex);\n      if (!canvasTextChangesForNode) {\n        canvasTextChangesForNode = [];\n        canvasTextChanges.set(change.nodeIndex, canvasTextChangesForNode);\n      }\n\n      canvasTextChangesForNode.push(change);\n    }\n  }\n\n  for (const [nodeIndex, canvasTextChangesForNode] of canvasTextChanges.entries()) {\n    const node = canvasData.nodes[nodeIndex];\n    if (!node) {\n      console.warn('Node not found', {\n        nodeIndex,\n        path\n      });\n\n      return null;\n    }\n\n    if (typeof node.text !== 'string') {\n      console.warn('Node text is not a string', {\n        nodeIndex,\n        path\n      });\n\n      return null;\n    }\n\n    const contentChanges = canvasTextChangesForNode.map((change) => referenceToFileChange(change.originalReference, change.newContent));\n    node.text = await applyContentChanges(node.text, `${path}.node${nodeIndex.toString()}.VIRTUAL_FILE.md`, contentChanges);\n  }\n\n  return JSON.stringify(canvasData, null, '\\t');\n}\n\nfunction parseJsonSafe(content: string): Record<string, unknown> {\n  let parsed: unknown;\n  try {\n    parsed = JSON.parse(content);\n  } catch {\n    parsed = null;\n  }\n\n  if (parsed === null || typeof parsed !== 'object') {\n    parsed = {};\n  }\n\n  return parsed as Record<string, unknown>;\n}\n"],
  "mappings": "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",
  "names": []
}

|
@@ -253,6 +253,39 @@ export interface UpdateLinksInFileOptions extends ProcessOptions {
|
|
253
253
|
*/
|
254
254
|
shouldUpdateFilenameAlias?: boolean;
|
255
255
|
}
|
256
|
+
/**
|
257
|
+
* The options for updating the links in a content string.
|
258
|
+
*/
|
259
|
+
interface UpdateLinksInContentOptions {
|
260
|
+
/**
|
261
|
+
* The Obsidian application instance.
|
262
|
+
*/
|
263
|
+
app: App;
|
264
|
+
/**
|
265
|
+
* The content to update the links in.
|
266
|
+
*/
|
267
|
+
content: string;
|
268
|
+
/**
|
269
|
+
* The new source path or file.
|
270
|
+
*/
|
271
|
+
newSourcePathOrFile: PathOrFile;
|
272
|
+
/**
|
273
|
+
* The old source path or file.
|
274
|
+
*/
|
275
|
+
oldSourcePathOrFile?: PathOrFile;
|
276
|
+
/**
|
277
|
+
* Whether to force markdown links.
|
278
|
+
*/
|
279
|
+
shouldForceMarkdownLinks?: boolean;
|
280
|
+
/**
|
281
|
+
* Whether to update only embedded links.
|
282
|
+
*/
|
283
|
+
shouldUpdateEmbedOnlyLinks?: boolean;
|
284
|
+
/**
|
285
|
+
* Whether to update filename alias.
|
286
|
+
*/
|
287
|
+
shouldUpdateFilenameAlias?: boolean;
|
288
|
+
}
|
256
289
|
/**
|
257
290
|
* Converts a link to a new path.
|
258
291
|
*
|
@@ -280,6 +313,15 @@ export declare function editBacklinks(app: App, pathOrFile: PathOrFile, linkConv
|
|
280
313
|
* @returns A promise that resolves when the backlinks have been edited.
|
281
314
|
*/
|
282
315
|
export declare function editLinks(app: App, pathOrFile: PathOrFile, linkConverter: (link: Reference) => MaybePromise<string | void>, processOptions?: ProcessOptions): Promise<void>;
|
316
|
+
/**
|
317
|
+
* Edits the links in a content string.
|
318
|
+
*
|
319
|
+
* @param app - The Obsidian application instance.
|
320
|
+
* @param content - The content to edit the links in.
|
321
|
+
* @param linkConverter - The function that converts each link.
|
322
|
+
* @returns A promise that resolves when the links are updated.
|
323
|
+
*/
|
324
|
+
export declare function editLinksInContent(app: App, content: string, linkConverter: (link: Reference) => MaybePromise<string | void>): Promise<void>;
|
283
325
|
/**
|
284
326
|
* Extracts the file associated with a link.
|
285
327
|
*
|
@@ -364,6 +406,13 @@ export declare function testWikilink(link: string): boolean;
|
|
364
406
|
* @returns The updated link.
|
365
407
|
*/
|
366
408
|
export declare function updateLink(options: UpdateLinkOptions): string;
|
409
|
+
/**
|
410
|
+
* Updates the links in a content string based on the provided parameters.
|
411
|
+
*
|
412
|
+
* @param options - The options for updating the links.
|
413
|
+
* @returns A promise that resolves when the links are updated.
|
414
|
+
*/
|
415
|
+
export declare function updateLinksInContent(options: UpdateLinksInContentOptions): Promise<void>;
|
367
416
|
/**
|
368
417
|
* Updates the links in a file based on the provided parameters.
|
369
418
|
*
|
@@ -371,3 +420,4 @@ export declare function updateLink(options: UpdateLinkOptions): string;
|
|
371
420
|
* @returns A promise that resolves when the links are updated.
|
372
421
|
*/
|
373
422
|
export declare function updateLinksInFile(options: UpdateLinksInFileOptions): Promise<void>;
|
423
|
+
export {};
|