obsidian-dev-utils
Version:
This is the collection of useful functions that you can use for your Obsidian plugin development
379 lines (375 loc) • 48.5 kB
JavaScript
/*
THIS IS A GENERATED/BUNDLED FILE BY ESBUILD
if you want to view the source, please visit the github repository of this plugin
*/
(function initEsm() {
if (globalThis.process) {
return;
}
const browserProcess = {
browser: true,
cwd() {
return '/';
},
env: {},
platform: 'android'
};
globalThis.process = browserProcess;
})();
import {
isFrontmatterLinkCache,
isReferenceCache
} from "obsidian-typings/implementations";
import { getLibDebugger } from "../Debug.mjs";
import { printError } from "../Error.mjs";
import {
deepEqual,
getNestedPropertyValue,
setNestedPropertyValue
} from "../ObjectUtils.mjs";
import { resolveValue } from "../ValueProvider.mjs";
import {
getPath,
isCanvasFile
} from "./FileSystem.mjs";
import {
parseFrontmatter,
setFrontmatter
} from "./Frontmatter.mjs";
import {
isFrontmatterLinkCacheWithOffsets,
toFrontmatterLinkCacheWithOffsets
} from "./FrontmatterLinkCacheWithOffsets.mjs";
import {
isCanvasReference,
referenceToFileChange
} from "./Reference.mjs";
import { process } from "./Vault.mjs";
async function applyContentChanges(abortSignal, content, path, changesProvider, shouldRetryOnInvalidChanges = true) {
abortSignal.throwIfAborted();
let changes = await resolveValue(changesProvider, abortSignal, content);
abortSignal.throwIfAborted();
if (changes === null) {
return null;
}
const { frontmatter, hasFrontmatterError } = parseFrontmatterSafely(content, path);
if (!validateChanges(changes, content, frontmatter, path)) {
return shouldRetryOnInvalidChanges ? null : content;
}
changes = sortAndFilterChanges(changes);
const { frontmatterChanged, newContent } = applyContentChangesToText(changes, content, hasFrontmatterError, path);
await applyFrontmatterChangesWithOffsets(abortSignal, frontmatter, frontmatterChanged, path);
abortSignal.throwIfAborted();
return buildFinalContent(newContent, frontmatter, frontmatterChanged);
}
async function applyFileChanges(app, pathOrFile, changesProvider, processOptions = {}, shouldRetryOnInvalidChanges = true) {
await process(app, pathOrFile, async (abortSignal, content) => {
if (isCanvasFile(app, pathOrFile)) {
return await applyCanvasChanges(abortSignal, content, getPath(app, pathOrFile), changesProvider, shouldRetryOnInvalidChanges);
}
return await applyContentChanges(abortSignal, content, getPath(app, pathOrFile), changesProvider, shouldRetryOnInvalidChanges);
}, processOptions);
}
function isCanvasChange(change) {
return isCanvasReference(change.reference);
}
function isCanvasFileNodeChange(change) {
return isCanvasChange(change) && change.reference.type === "file";
}
function isCanvasTextNodeChange(change) {
return isCanvasChange(change) && change.reference.type === "text";
}
function isContentChange(fileChange) {
return isReferenceCache(fileChange.reference);
}
function isFrontmatterChange(fileChange) {
return isFrontmatterLinkCache(fileChange.reference);
}
function isFrontmatterChangeWithOffsets(fileChange) {
return isFrontmatterLinkCacheWithOffsets(fileChange.reference);
}
function toFrontmatterChangeWithOffsets(fileChange) {
if (isFrontmatterChangeWithOffsets(fileChange)) {
return fileChange;
}
return {
...fileChange,
reference: toFrontmatterLinkCacheWithOffsets(fileChange.reference)
};
}
async function applyCanvasChanges(abortSignal, content, path, changesProvider, shouldRetryOnInvalidChanges = true) {
const changes = await resolveValue(changesProvider, abortSignal, content);
abortSignal.throwIfAborted();
if (changes === null) {
return null;
}
const canvasData = parseJsonSafe(content);
const canvasTextChanges = /* @__PURE__ */ new Map();
for (const change of changes) {
if (!isCanvasChange(change)) {
const message = "Only canvas changes are supported for canvas files";
console.error(message, {
change,
path
});
continue;
}
const node = canvasData.nodes[change.reference.nodeIndex];
if (!node) {
const message = "Node not found";
console.error(message, {
nodeIndex: change.reference.nodeIndex,
path
});
return null;
}
if (isCanvasFileNodeChange(change)) {
if (node.file !== change.oldContent) {
getLibDebugger("FileChange:applyCanvasChanges")("Content mismatch", {
actualContent: node.file,
expectedContent: change.oldContent,
nodeIndex: change.reference.nodeIndex,
path,
type: "file"
});
return null;
}
node.file = change.newContent;
} else if (isCanvasTextNodeChange(change)) {
let canvasTextChangesForNode = canvasTextChanges.get(change.reference.nodeIndex);
if (!canvasTextChangesForNode) {
canvasTextChangesForNode = [];
canvasTextChanges.set(change.reference.nodeIndex, canvasTextChangesForNode);
}
canvasTextChangesForNode.push(change);
}
}
for (const [nodeIndex, canvasTextChangesForNode] of canvasTextChanges.entries()) {
const node = canvasData.nodes[nodeIndex];
if (!node) {
const message = "Node not found";
console.error(message, {
nodeIndex,
path
});
return null;
}
if (typeof node.text !== "string") {
const message = "Node text is not a string";
console.error(message, {
nodeIndex,
path
});
return null;
}
const contentChanges = canvasTextChangesForNode.map((change) => referenceToFileChange(change.reference.originalReference, change.newContent));
node.text = await applyContentChanges(
abortSignal,
node.text,
`${path}.node${String(nodeIndex)}.VIRTUAL_FILE.md`,
contentChanges,
shouldRetryOnInvalidChanges
);
}
return JSON.stringify(canvasData, null, " ");
}
function applyContentChangesToText(changes, content, hasFrontmatterError, path) {
let newContent = "";
let lastIndex = 0;
let lastContentChange = {
newContent: "",
oldContent: "",
reference: {
link: "",
original: "",
position: {
end: { col: 0, line: 0, offset: 0 },
start: { col: 0, line: 0, offset: 0 }
}
}
};
const frontmatterChangesWithOffsetMap = /* @__PURE__ */ new Map();
for (const change of changes) {
if (isContentChange(change)) {
if (lastIndex <= change.reference.position.start.offset) {
newContent += content.slice(lastIndex, change.reference.position.start.offset);
newContent += change.newContent;
lastIndex = change.reference.position.end.offset;
lastContentChange = change;
} else {
const overlappingStartOffset = change.reference.position.start.offset - lastContentChange.reference.position.start.offset;
const overlappingEndOffset = change.reference.position.end.offset - lastContentChange.reference.position.start.offset;
const overlappingContent = lastContentChange.newContent.slice(overlappingStartOffset, overlappingEndOffset);
if (overlappingContent !== change.oldContent) {
const message = "Overlapping changes";
console.error(message, { change, lastContentChange });
throw new Error(message);
}
newContent = newContent.slice(0, newContent.length - lastContentChange.newContent.length) + lastContentChange.newContent.slice(0, overlappingStartOffset) + change.newContent + lastContentChange.newContent.slice(overlappingEndOffset);
}
} else if (isFrontmatterChange(change)) {
if (hasFrontmatterError) {
console.error(`Cannot apply frontmatter change in ${path}, because frontmatter parsing failed`, { change });
} else {
let frontmatterChangesWithOffsets = frontmatterChangesWithOffsetMap.get(change.reference.key);
if (!frontmatterChangesWithOffsets) {
frontmatterChangesWithOffsets = [];
frontmatterChangesWithOffsetMap.set(change.reference.key, frontmatterChangesWithOffsets);
}
frontmatterChangesWithOffsets.push(toFrontmatterChangeWithOffsets(change));
}
}
}
newContent += content.slice(lastIndex);
return { frontmatterChanged: frontmatterChangesWithOffsetMap, newContent };
}
async function applyFrontmatterChangesWithOffsets(abortSignal, frontmatter, frontmatterChangesWithOffsetMap, path) {
for (const [key, frontmatterChangesWithOffsets] of frontmatterChangesWithOffsetMap.entries()) {
const propertyValue = getNestedPropertyValue(frontmatter, key);
if (typeof propertyValue !== "string") {
return;
}
const contentChanges = frontmatterChangesWithOffsets.map((change) => ({
newContent: change.newContent,
oldContent: change.oldContent,
reference: {
link: "",
original: "",
position: {
end: {
col: change.reference.endOffset,
line: 0,
offset: change.reference.endOffset
},
start: {
col: change.reference.startOffset,
line: 0,
offset: change.reference.startOffset
}
}
}
}));
const newPropertyValue = await applyContentChanges(abortSignal, propertyValue, `${path}.frontmatter.${key}.VIRTUAL_FILE.md`, contentChanges);
if (newPropertyValue === null) {
return;
}
setNestedPropertyValue(frontmatter, key, newPropertyValue);
}
}
function buildFinalContent(newContent, frontmatter, frontmatterChanged) {
if (frontmatterChanged.size > 0) {
return setFrontmatter(newContent, frontmatter);
}
return newContent;
}
function parseFrontmatterSafely(content, path) {
let frontmatter = {};
let hasFrontmatterError = false;
try {
frontmatter = parseFrontmatter(content);
} catch (error) {
printError(new Error(`Frontmatter parsing failed in ${path}`, { cause: error }));
hasFrontmatterError = true;
}
return { frontmatter, hasFrontmatterError };
}
function parseJsonSafe(content) {
let parsed;
try {
parsed = JSON.parse(content);
} catch {
parsed = null;
}
if (parsed === null || typeof parsed !== "object") {
parsed = {};
}
return parsed;
}
function sortAndFilterChanges(changes) {
changes.sort((a, b) => {
if (isContentChange(a) && isContentChange(b)) {
return a.reference.position.start.offset - b.reference.position.start.offset;
}
if (isFrontmatterChangeWithOffsets(a) && isFrontmatterChangeWithOffsets(b)) {
return a.reference.key.localeCompare(b.reference.key) || a.reference.startOffset - b.reference.startOffset;
}
if (isFrontmatterChange(a) && isFrontmatterChange(b)) {
return a.reference.key.localeCompare(b.reference.key);
}
return isContentChange(a) ? -1 : 1;
});
return changes.filter((change, index) => {
if (change.oldContent === change.newContent) {
return false;
}
if (index === 0) {
return true;
}
return !deepEqual(change, changes[index - 1]);
});
}
function validateChanges(changes, content, frontmatter, path) {
const validateChangesDebugger = getLibDebugger("FileChange:validateChanges");
for (const change of changes) {
if (isContentChange(change)) {
const startOffset = change.reference.position.start.offset;
const endOffset = change.reference.position.end.offset;
const actualContent = content.slice(startOffset, endOffset);
if (actualContent !== change.oldContent) {
validateChangesDebugger("Content mismatch", {
actualContent,
endOffset,
expectedContent: change.oldContent,
path,
startOffset
});
return false;
}
} else if (isFrontmatterChangeWithOffsets(change)) {
const propertyValue = getNestedPropertyValue(frontmatter, change.reference.key);
if (typeof propertyValue !== "string") {
validateChangesDebugger("Property value is not a string", {
frontmatterKey: change.reference.key,
path,
propertyValue
});
return false;
}
const actualContent = propertyValue.slice(change.reference.startOffset, change.reference.endOffset);
if (actualContent !== change.oldContent) {
validateChangesDebugger("Content mismatch", {
actualContent,
expectedContent: change.oldContent,
frontmatterKey: change.reference.key,
path,
startOffset: change.reference.startOffset
});
return false;
}
} else if (isFrontmatterChange(change)) {
const actualContent = getNestedPropertyValue(frontmatter, change.reference.key);
if (actualContent !== change.oldContent) {
validateChangesDebugger("Content mismatch", {
actualContent,
expectedContent: change.oldContent,
frontmatterKey: change.reference.key,
path
});
return false;
}
}
}
return true;
}
export {
applyContentChanges,
applyFileChanges,
isCanvasChange,
isCanvasFileNodeChange,
isCanvasTextNodeChange,
isContentChange,
isFrontmatterChange,
isFrontmatterChangeWithOffsets,
toFrontmatterChangeWithOffsets
};
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/FileChange.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility types and functions for handling file changes in Obsidian.\n */\n\nimport type {\n  App,\n  FrontmatterLinkCache,\n  Reference,\n  ReferenceCache\n} from 'obsidian';\nimport type { CanvasData } from 'obsidian/Canvas.d.ts';\n\nimport {\n  isFrontmatterLinkCache,\n  isReferenceCache\n} from 'obsidian-typings/implementations';\n\nimport type { GenericObject } from '../ObjectUtils.ts';\nimport type { ValueProvider } from '../ValueProvider.ts';\nimport type { PathOrFile } from './FileSystem.ts';\nimport type { CombinedFrontmatter } from './Frontmatter.ts';\nimport type { FrontmatterLinkCacheWithOffsets } from './FrontmatterLinkCacheWithOffsets.ts';\nimport type {\n  CanvasFileNodeReference,\n  CanvasReference,\n  CanvasTextNodeReference\n} from './Reference.ts';\nimport type { ProcessOptions } from './Vault.ts';\n\nimport { getLibDebugger } from '../Debug.ts';\nimport { printError } from '../Error.ts';\nimport {\n  deepEqual,\n  getNestedPropertyValue,\n  setNestedPropertyValue\n} from '../ObjectUtils.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 {\n  isFrontmatterLinkCacheWithOffsets,\n  toFrontmatterLinkCacheWithOffsets\n} from './FrontmatterLinkCacheWithOffsets.ts';\nimport {\n  isCanvasReference,\n  referenceToFileChange\n} from './Reference.ts';\nimport { process } from './Vault.ts';\n\n/**\n * A file change in the vault.\n */\nexport interface FileChange {\n  /**\n   * A new content to replace the old content.\n   */\n  newContent: string;\n\n  /**\n   * An old content that will be replaced.\n   */\n  oldContent: string;\n\n  /**\n   * A reference that caused the change.\n   */\n  reference: Reference;\n}\ntype CanvasChange = { reference: CanvasReference } & FileChange;\ntype CanvasFileNodeChange = { reference: CanvasFileNodeReference } & FileChange;\ntype CanvasTextNodeChange = { reference: CanvasTextNodeReference } & FileChange;\ntype ContentChange = { reference: ReferenceCache } & FileChange;\ntype FrontmatterChange = { reference: FrontmatterLinkCache } & FileChange;\ntype FrontmatterChangeWithOffsets = { reference: FrontmatterLinkCacheWithOffsets } & FileChange;\n\n/**\n * Applies a series of content changes to the specified content.\n *\n * @param abortSignal - The abort signal to control the execution of the function.\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 * @param shouldRetryOnInvalidChanges - Whether to retry the operation if the changes are invalid.\n * @returns A {@link Promise} that resolves to the updated content or to `null` if update didn't succeed.\n */\nexport async function applyContentChanges(\n  abortSignal: AbortSignal,\n  content: string,\n  path: string,\n  changesProvider: ValueProvider<FileChange[] | null, [content: string]>,\n  shouldRetryOnInvalidChanges = true\n): Promise<null | string> {\n  abortSignal.throwIfAborted();\n  let changes = await resolveValue(changesProvider, abortSignal, content);\n  abortSignal.throwIfAborted();\n  if (changes === null) {\n    return null;\n  }\n\n  const { frontmatter, hasFrontmatterError } = parseFrontmatterSafely(content, path);\n\n  if (!validateChanges(changes, content, frontmatter, path)) {\n    return shouldRetryOnInvalidChanges ? null : content;\n  }\n\n  changes = sortAndFilterChanges(changes);\n\n  const { frontmatterChanged, newContent } = applyContentChangesToText(changes, content, hasFrontmatterError, path);\n\n  await applyFrontmatterChangesWithOffsets(abortSignal, frontmatter, frontmatterChanged, path);\n  abortSignal.throwIfAborted();\n\n  return buildFinalContent(newContent, frontmatter, frontmatterChanged);\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 * @param shouldRetryOnInvalidChanges - Whether to retry the operation if the changes are invalid.\n *\n * @returns A {@link 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[] | null, [content: string]>,\n  processOptions: ProcessOptions = {},\n  shouldRetryOnInvalidChanges = true\n): Promise<void> {\n  await process(app, pathOrFile, async (abortSignal, content) => {\n    if (isCanvasFile(app, pathOrFile)) {\n      return await applyCanvasChanges(abortSignal, content, getPath(app, pathOrFile), changesProvider, shouldRetryOnInvalidChanges);\n    }\n\n    return await applyContentChanges(abortSignal, content, getPath(app, pathOrFile), changesProvider, shouldRetryOnInvalidChanges);\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 isCanvasReference(change.reference);\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.reference.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.reference.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 isReferenceCache(fileChange.reference);\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 isFrontmatterLinkCache(fileChange.reference);\n}\n\n/**\n * Checks if a file change is a frontmatter change with offsets.\n *\n * @param fileChange - The file change to check.\n * @returns A boolean indicating whether the file change is a frontmatter change with offsets.\n */\nexport function isFrontmatterChangeWithOffsets(fileChange: FileChange): fileChange is FrontmatterChangeWithOffsets {\n  return isFrontmatterLinkCacheWithOffsets(fileChange.reference);\n}\n\n/**\n * Converts a frontmatter change to a frontmatter change with offsets.\n *\n * @param fileChange - The file change to convert.\n * @returns The converted file change.\n */\nexport function toFrontmatterChangeWithOffsets(fileChange: FrontmatterChange): FrontmatterChangeWithOffsets {\n  if (isFrontmatterChangeWithOffsets(fileChange)) {\n    return fileChange;\n  }\n\n  return {\n    ...fileChange,\n    reference: toFrontmatterLinkCacheWithOffsets(fileChange.reference)\n  };\n}\n\nasync function applyCanvasChanges(\n  abortSignal: AbortSignal,\n  content: string,\n  path: string,\n  changesProvider: ValueProvider<FileChange[] | null, [content: string]>,\n  shouldRetryOnInvalidChanges = true\n): Promise<null | string> {\n  const changes = await resolveValue(changesProvider, abortSignal, content);\n  abortSignal.throwIfAborted();\n  if (changes === null) {\n    return null;\n  }\n\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      const message = 'Only canvas changes are supported for canvas files';\n      console.error(message, {\n        change,\n        path\n      });\n      continue;\n    }\n\n    const node = canvasData.nodes[change.reference.nodeIndex];\n    if (!node) {\n      const message = 'Node not found';\n      console.error(message, {\n        nodeIndex: change.reference.nodeIndex,\n        path\n      });\n      return null;\n    }\n\n    if (isCanvasFileNodeChange(change)) {\n      if (node.file !== change.oldContent) {\n        getLibDebugger('FileChange:applyCanvasChanges')('Content mismatch', {\n          actualContent: node.file as string | undefined,\n          expectedContent: change.oldContent,\n          nodeIndex: change.reference.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.reference.nodeIndex);\n      if (!canvasTextChangesForNode) {\n        canvasTextChangesForNode = [];\n        canvasTextChanges.set(change.reference.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      const message = 'Node not found';\n      console.error(message, {\n        nodeIndex,\n        path\n      });\n\n      return null;\n    }\n\n    if (typeof node.text !== 'string') {\n      const message = 'Node text is not a string';\n      console.error(message, {\n        nodeIndex,\n        path\n      });\n\n      return null;\n    }\n\n    const contentChanges = canvasTextChangesForNode.map((change) => referenceToFileChange(change.reference.originalReference, change.newContent));\n    node.text = await applyContentChanges(\n      abortSignal,\n      node.text,\n      `${path}.node${String(nodeIndex)}.VIRTUAL_FILE.md`,\n      contentChanges,\n      shouldRetryOnInvalidChanges\n    );\n  }\n\n  return JSON.stringify(canvasData, null, '\\t');\n}\n\nfunction applyContentChangesToText(\n  changes: FileChange[],\n  content: string,\n  hasFrontmatterError: boolean,\n  path: string\n): { frontmatterChanged: Map<string, FrontmatterChangeWithOffsets[]>; newContent: string } {\n  let newContent = '';\n  let lastIndex = 0;\n  let lastContentChange: ContentChange = {\n    newContent: '',\n    oldContent: '',\n    reference: {\n      link: '',\n      original: '',\n      position: {\n        end: { col: 0, line: 0, offset: 0 },\n        start: { col: 0, line: 0, offset: 0 }\n      }\n    }\n  };\n  const frontmatterChangesWithOffsetMap = new Map<string, FrontmatterChangeWithOffsets[]>();\n\n  for (const change of changes) {\n    if (isContentChange(change)) {\n      if (lastIndex <= change.reference.position.start.offset) {\n        newContent += content.slice(lastIndex, change.reference.position.start.offset);\n        newContent += change.newContent;\n        lastIndex = change.reference.position.end.offset;\n        lastContentChange = change;\n      } else {\n        const overlappingStartOffset = change.reference.position.start.offset - lastContentChange.reference.position.start.offset;\n        const overlappingEndOffset = change.reference.position.end.offset - lastContentChange.reference.position.start.offset;\n        const overlappingContent = lastContentChange.newContent.slice(overlappingStartOffset, overlappingEndOffset);\n        if (overlappingContent !== change.oldContent) {\n          const message = 'Overlapping changes';\n          console.error(message, { change, lastContentChange });\n          throw new Error(message);\n        }\n        newContent = newContent.slice(0, newContent.length - lastContentChange.newContent.length)\n          + lastContentChange.newContent.slice(0, overlappingStartOffset)\n          + change.newContent\n          + lastContentChange.newContent.slice(overlappingEndOffset);\n      }\n    } else if (isFrontmatterChange(change)) {\n      if (hasFrontmatterError) {\n        console.error(`Cannot apply frontmatter change in ${path}, because frontmatter parsing failed`, { change });\n      } else {\n        let frontmatterChangesWithOffsets = frontmatterChangesWithOffsetMap.get(change.reference.key);\n        if (!frontmatterChangesWithOffsets) {\n          frontmatterChangesWithOffsets = [];\n          frontmatterChangesWithOffsetMap.set(change.reference.key, frontmatterChangesWithOffsets);\n        }\n        frontmatterChangesWithOffsets.push(toFrontmatterChangeWithOffsets(change));\n      }\n    }\n  }\n\n  newContent += content.slice(lastIndex);\n\n  return { frontmatterChanged: frontmatterChangesWithOffsetMap, newContent };\n}\n\nasync function applyFrontmatterChangesWithOffsets(\n  abortSignal: AbortSignal,\n  frontmatter: CombinedFrontmatter<unknown>,\n  frontmatterChangesWithOffsetMap: Map<string, FrontmatterChangeWithOffsets[]>,\n  path: string\n): Promise<void> {\n  for (const [key, frontmatterChangesWithOffsets] of frontmatterChangesWithOffsetMap.entries()) {\n    const propertyValue = getNestedPropertyValue(frontmatter, key);\n    if (typeof propertyValue !== 'string') {\n      return;\n    }\n\n    const contentChanges: ContentChange[] = frontmatterChangesWithOffsets.map((change) => ({\n      newContent: change.newContent,\n      oldContent: change.oldContent,\n      reference: {\n        link: '',\n        original: '',\n        position: {\n          end: {\n            col: change.reference.endOffset,\n            line: 0,\n            offset: change.reference.endOffset\n          },\n          start: {\n            col: change.reference.startOffset,\n            line: 0,\n            offset: change.reference.startOffset\n          }\n        }\n      }\n    } as ContentChange));\n\n    const newPropertyValue = await applyContentChanges(abortSignal, propertyValue, `${path}.frontmatter.${key}.VIRTUAL_FILE.md`, contentChanges);\n    if (newPropertyValue === null) {\n      return;\n    }\n\n    setNestedPropertyValue(frontmatter, key, newPropertyValue);\n  }\n}\n\nfunction buildFinalContent(\n  newContent: string,\n  frontmatter: CombinedFrontmatter<unknown>,\n  frontmatterChanged: Map<string, FrontmatterChangeWithOffsets[]>\n): string {\n  if (frontmatterChanged.size > 0) {\n    return setFrontmatter(newContent, frontmatter);\n  }\n  return newContent;\n}\n\nfunction parseFrontmatterSafely(content: string, path: string): { frontmatter: CombinedFrontmatter<unknown>; hasFrontmatterError: boolean } {\n  let frontmatter: CombinedFrontmatter<unknown> = {};\n  let hasFrontmatterError = false;\n\n  try {\n    frontmatter = parseFrontmatter(content);\n  } catch (error) {\n    printError(new Error(`Frontmatter parsing failed in ${path}`, { cause: error }));\n    hasFrontmatterError = true;\n  }\n\n  return { frontmatter, hasFrontmatterError };\n}\n\nfunction parseJsonSafe(content: string): GenericObject {\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 GenericObject;\n}\n\nfunction sortAndFilterChanges(changes: FileChange[]): FileChange[] {\n  // Sort changes by type and position\n  changes.sort((a, b) => {\n    if (isContentChange(a) && isContentChange(b)) {\n      return a.reference.position.start.offset - b.reference.position.start.offset;\n    }\n\n    if (isFrontmatterChangeWithOffsets(a) && isFrontmatterChangeWithOffsets(b)) {\n      return a.reference.key.localeCompare(b.reference.key) || a.reference.startOffset - b.reference.startOffset;\n    }\n\n    if (isFrontmatterChange(a) && isFrontmatterChange(b)) {\n      return a.reference.key.localeCompare(b.reference.key);\n    }\n\n    return isContentChange(a) ? -1 : 1;\n  });\n\n  // Filter out duplicate and no-op changes\n  return 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\nfunction validateChanges(changes: FileChange[], content: string, frontmatter: CombinedFrontmatter<unknown>, path: string): boolean {\n  const validateChangesDebugger = getLibDebugger('FileChange:validateChanges');\n  for (const change of changes) {\n    if (isContentChange(change)) {\n      const startOffset = change.reference.position.start.offset;\n      const endOffset = change.reference.position.end.offset;\n      const actualContent = content.slice(startOffset, endOffset);\n      if (actualContent !== change.oldContent) {\n        validateChangesDebugger('Content mismatch', {\n          actualContent,\n          endOffset,\n          expectedContent: change.oldContent,\n          path,\n          startOffset\n        });\n\n        return false;\n      }\n    } else if (isFrontmatterChangeWithOffsets(change)) {\n      const propertyValue = getNestedPropertyValue(frontmatter, change.reference.key);\n      if (typeof propertyValue !== 'string') {\n        validateChangesDebugger('Property value is not a string', {\n          frontmatterKey: change.reference.key,\n          path,\n          propertyValue\n        });\n        return false;\n      }\n\n      const actualContent = propertyValue.slice(change.reference.startOffset, change.reference.endOffset);\n      if (actualContent !== change.oldContent) {\n        validateChangesDebugger('Content mismatch', {\n          actualContent,\n          expectedContent: change.oldContent,\n          frontmatterKey: change.reference.key,\n          path,\n          startOffset: change.reference.startOffset\n        });\n\n        return false;\n      }\n    } else if (isFrontmatterChange(change)) {\n      const actualContent = getNestedPropertyValue(frontmatter, change.reference.key);\n      if (actualContent !== change.oldContent) {\n        validateChangesDebugger('Content mismatch', {\n          actualContent,\n          expectedContent: change.oldContent,\n          frontmatterKey: change.reference.key,\n          path\n        });\n\n        return false;\n      }\n    }\n  }\n\n  return true;\n}\n"],
  "mappings": "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",
  "names": []
}
