obsidian-dev-utils
Version:
This is the collection of useful functions that you can use for your Obsidian plugin development
338 lines (335 loc) • 44.4 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 initCjs(){const globalThisRecord=globalThis;globalThisRecord["__name"]??=name;const originalRequire=require;if(originalRequire&&!originalRequire.__isPatched){require=Object.assign(id=>requirePatched(id),originalRequire,{__isPatched:true})}const newFuncs={__extractDefault:__name(()=>extractDefault,"__extractDefault"),process:__name(()=>{const browserProcess={browser:true,cwd:__name(()=>"/","cwd"),env:{},platform:"android"};return browserProcess},"process")};for(const key of Object.keys(newFuncs)){globalThisRecord[key]??=newFuncs[key]?.()}function name(obj){return obj}__name(name,"name");function extractDefault(module){return module&&module.__esModule&&"default"in module?module.default:module}__name(extractDefault,"extractDefault");function requirePatched(id){const module=originalRequire?.(id);if(module){return extractDefault(module)}if(id==="process"||id==="node:process"){console.error(`Module not found: ${id}. Fake process object is returned instead.`);return globalThis.process}console.error(`Module not found: ${id}. Empty object is returned instead.`);return{}}__name(requirePatched,"requirePatched")})();
;
var __defProp = Object.defineProperty;
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
var __getOwnPropNames = Object.getOwnPropertyNames;
var __hasOwnProp = Object.prototype.hasOwnProperty;
var __export = (target, all) => {
for (var name in all)
__defProp(target, name, { get: all[name], enumerable: true });
};
var __copyProps = (to, from, except, desc) => {
if (from && typeof from === "object" || typeof from === "function") {
for (let key of __getOwnPropNames(from))
if (!__hasOwnProp.call(to, key) && key !== except)
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
}
return to;
};
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
var FileChange_exports = {};
__export(FileChange_exports, {
applyContentChanges: () => applyContentChanges,
applyFileChanges: () => applyFileChanges,
isCanvasChange: () => isCanvasChange,
isCanvasFileNodeChange: () => isCanvasFileNodeChange,
isCanvasTextNodeChange: () => isCanvasTextNodeChange,
isContentChange: () => isContentChange,
isFrontmatterChange: () => isFrontmatterChange,
isFrontmatterChangeWithOffsets: () => isFrontmatterChangeWithOffsets
});
module.exports = __toCommonJS(FileChange_exports);
var import_implementations = require('obsidian-typings/implementations');
var import_Debug = require('../Debug.cjs');
var import_ObjectUtils = require('../ObjectUtils.cjs');
var import_ValueProvider = require('../ValueProvider.cjs');
var import_FileSystem = require('./FileSystem.cjs');
var import_Frontmatter = require('./Frontmatter.cjs');
var import_FrontmatterLinkCacheWithOffsets = require('./FrontmatterLinkCacheWithOffsets.cjs');
var import_Reference = require('./Reference.cjs');
var import_Vault = require('./Vault.cjs');
async function applyContentChanges(content, path, changesProvider, shouldRetryOnInvalidChanges = true) {
let changes = await (0, import_ValueProvider.resolveValue)(changesProvider);
let frontmatter = {};
let hasFrontmatterError = false;
try {
frontmatter = (0, import_Frontmatter.parseFrontmatter)(content);
} catch (error) {
console.error(new Error(`Frontmatter parsing failed in ${path}`, { cause: error }));
hasFrontmatterError = true;
}
if (!validateChanges(changes, content, frontmatter, path, shouldRetryOnInvalidChanges)) {
return shouldRetryOnInvalidChanges ? null : content;
}
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;
});
changes = changes.filter((change, index) => {
if (change.oldContent === change.newContent) {
return false;
}
if (index === 0) {
return true;
}
return !(0, import_ObjectUtils.deepEqual)(change, changes[index - 1]);
});
for (let i = 1; i < changes.length; i++) {
const change = changes[i];
if (!change) {
continue;
}
const previousChange = changes[i - 1];
if (!previousChange) {
continue;
}
if (isContentChange(previousChange) && isContentChange(change) && previousChange.reference.position.end.offset && previousChange.reference.position.end.offset > change.reference.position.start.offset) {
console.warn("Overlapping changes", {
change,
previousChange
});
return null;
}
}
let newContent = "";
let lastIndex = 0;
let frontmatterChanged = false;
const frontmatterChangesWithOffsetMap = /* @__PURE__ */ new Map();
for (const change of changes) {
if (isContentChange(change)) {
newContent += content.slice(lastIndex, change.reference.position.start.offset);
newContent += change.newContent;
lastIndex = change.reference.position.end.offset;
} else if (isFrontmatterChangeWithOffsets(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(change);
frontmatterChanged = true;
}
} else if (isFrontmatterChange(change)) {
if (hasFrontmatterError) {
console.error(`Cannot apply frontmatter change in ${path}, because frontmatter parsing failed`, {
change
});
} else {
(0, import_ObjectUtils.setNestedPropertyValue)(frontmatter, change.reference.key, change.newContent);
frontmatterChanged = true;
}
}
}
await applyFrontmatterChangesWithOffsets(frontmatter, frontmatterChangesWithOffsetMap, path);
newContent += content.slice(lastIndex);
if (frontmatterChanged) {
newContent = (0, import_Frontmatter.setFrontmatter)(newContent, frontmatter);
}
return newContent;
}
async function applyFileChanges(app, pathOrFile, changesProvider, processOptions = {}, shouldRetryOnInvalidChanges = true) {
await (0, import_Vault.process)(app, pathOrFile, async (content) => {
if ((0, import_FileSystem.isCanvasFile)(app, pathOrFile)) {
return applyCanvasChanges(content, (0, import_FileSystem.getPath)(app, pathOrFile), changesProvider, shouldRetryOnInvalidChanges);
}
return await applyContentChanges(content, (0, import_FileSystem.getPath)(app, pathOrFile), changesProvider, shouldRetryOnInvalidChanges);
}, processOptions);
}
function isCanvasChange(change) {
return (0, import_Reference.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 (0, import_implementations.isReferenceCache)(fileChange.reference);
}
function isFrontmatterChange(fileChange) {
return (0, import_implementations.isFrontmatterLinkCache)(fileChange.reference);
}
function isFrontmatterChangeWithOffsets(fileChange) {
return (0, import_FrontmatterLinkCacheWithOffsets.isFrontmatterLinkCacheWithOffsets)(fileChange.reference);
}
async function applyCanvasChanges(content, path, changesProvider, shouldRetryOnInvalidChanges = true) {
const changes = await (0, import_ValueProvider.resolveValue)(changesProvider);
const canvasData = parseJsonSafe(content);
const canvasTextChanges = /* @__PURE__ */ new Map();
for (const change of changes) {
if (!isCanvasChange(change)) {
console.warn("Only canvas changes are supported for canvas files", {
change,
path
});
return null;
}
const node = canvasData.nodes[change.reference.nodeIndex];
if (!node) {
console.warn("Node not found", {
nodeIndex: change.reference.nodeIndex,
path
});
return null;
}
if (isCanvasFileNodeChange(change)) {
if (node.file !== change.oldContent) {
console.warn("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) {
console.warn("Node not found", {
nodeIndex,
path
});
return null;
}
if (typeof node.text !== "string") {
console.warn("Node text is not a string", {
nodeIndex,
path
});
return null;
}
const contentChanges = canvasTextChangesForNode.map((change) => (0, import_Reference.referenceToFileChange)(change.reference.originalReference, change.newContent));
node.text = await applyContentChanges(node.text, `${path}.node${String(nodeIndex)}.VIRTUAL_FILE.md`, contentChanges, shouldRetryOnInvalidChanges);
}
return JSON.stringify(canvasData, null, " ");
}
async function applyFrontmatterChangesWithOffsets(frontmatter, frontmatterChangesWithOffsetMap, path) {
for (const [key, frontmatterChangesWithOffsets] of frontmatterChangesWithOffsetMap.entries()) {
const propertyValue = (0, import_ObjectUtils.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(propertyValue, `${path}.frontmatter.${key}.VIRTUAL_FILE.md`, contentChanges);
if (newPropertyValue === null) {
return;
}
(0, import_ObjectUtils.setNestedPropertyValue)(frontmatter, key, newPropertyValue);
}
}
function parseJsonSafe(content) {
let parsed;
try {
parsed = JSON.parse(content);
} catch {
parsed = null;
}
if (parsed === null || typeof parsed !== "object") {
parsed = {};
}
return parsed;
}
function validateChanges(changes, content, frontmatter, path, shouldShowWarning) {
const _debugger = (0, import_Debug.getDebugger)("validateChanges");
const logger = shouldShowWarning ? console.warn.bind(console) : _debugger;
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) {
logger("Content mismatch", {
actualContent,
endOffset,
expectedContent: change.oldContent,
path,
startOffset
});
return false;
}
} else if (isFrontmatterChangeWithOffsets(change)) {
const propertyValue = (0, import_ObjectUtils.getNestedPropertyValue)(frontmatter, change.reference.key);
if (typeof propertyValue !== "string") {
logger("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) {
logger("Content mismatch", {
actualContent,
expectedContent: change.oldContent,
frontmatterKey: change.reference.key,
path,
startOffset: change.reference.startOffset
});
return false;
}
} else if (isFrontmatterChange(change)) {
const actualContent = (0, import_ObjectUtils.getNestedPropertyValue)(frontmatter, change.reference.key);
if (actualContent !== change.oldContent) {
logger("Content mismatch", {
actualContent,
expectedContent: change.oldContent,
frontmatterKey: change.reference.key,
path
});
return false;
}
}
}
return true;
}
// Annotate the CommonJS export names for ESM import in node:
0 && (module.exports = {
applyContentChanges,
applyFileChanges,
isCanvasChange,
isCanvasFileNodeChange,
isCanvasTextNodeChange,
isContentChange,
isFrontmatterChange,
isFrontmatterChangeWithOffsets
});
//# 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 { getDebugger } from '../Debug.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 { isFrontmatterLinkCacheWithOffsets } from './FrontmatterLinkCacheWithOffsets.ts';\nimport {\n  isCanvasReference,\n  referenceToFileChange\n} from './Reference.ts';\nimport { process } from './Vault.ts';\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   * The 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 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  content: string,\n  path: string,\n  changesProvider: ValueProvider<FileChange[]>,\n  shouldRetryOnInvalidChanges = true\n): Promise<null | string> {\n  let changes = await resolveValue(changesProvider);\n  let frontmatter: CombinedFrontmatter<unknown> = {};\n  let hasFrontmatterError = false;\n  try {\n    frontmatter = parseFrontmatter(content);\n  } catch (error) {\n    console.error(new Error(`Frontmatter parsing failed in ${path}`, { cause: error }));\n    hasFrontmatterError = true;\n  }\n\n  if (!validateChanges(changes, content, frontmatter, path, shouldRetryOnInvalidChanges)) {\n    return shouldRetryOnInvalidChanges ? null : content;\n  }\n\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  // 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.reference.position.end.offset\n      && previousChange.reference.position.end.offset > change.reference.position.start.offset\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  const frontmatterChangesWithOffsetMap = new Map<string, FrontmatterChangeWithOffsets[]>();\n\n  for (const change of changes) {\n    if (isContentChange(change)) {\n      newContent += content.slice(lastIndex, change.reference.position.start.offset);\n      newContent += change.newContent;\n      lastIndex = change.reference.position.end.offset;\n    } else if (isFrontmatterChangeWithOffsets(change)) {\n      if (hasFrontmatterError) {\n        console.error(`Cannot apply frontmatter change in ${path}, because frontmatter parsing failed`, {\n          change\n        });\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(change);\n        frontmatterChanged = true;\n      }\n    } else if (isFrontmatterChange(change)) {\n      if (hasFrontmatterError) {\n        console.error(`Cannot apply frontmatter change in ${path}, because frontmatter parsing failed`, {\n          change\n        });\n      } else {\n        setNestedPropertyValue(frontmatter, change.reference.key, change.newContent);\n        frontmatterChanged = true;\n      }\n    }\n  }\n\n  await applyFrontmatterChangesWithOffsets(frontmatter, frontmatterChangesWithOffsetMap, path);\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 * @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[]>,\n  processOptions: ProcessOptions = {},\n  shouldRetryOnInvalidChanges = true\n): Promise<void> {\n  await process(app, pathOrFile, async (content) => {\n    if (isCanvasFile(app, pathOrFile)) {\n      return applyCanvasChanges(content, getPath(app, pathOrFile), changesProvider, shouldRetryOnInvalidChanges);\n    }\n\n    return await applyContentChanges(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\nasync function applyCanvasChanges(\n  content: string,\n  path: string,\n  changesProvider: ValueProvider<FileChange[]>,\n  shouldRetryOnInvalidChanges = true\n): 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.reference.nodeIndex];\n    if (!node) {\n      console.warn('Node not found', {\n        nodeIndex: change.reference.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.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      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.reference.originalReference, change.newContent));\n    node.text = await applyContentChanges(node.text, `${path}.node${String(nodeIndex)}.VIRTUAL_FILE.md`, contentChanges, shouldRetryOnInvalidChanges);\n  }\n\n  return JSON.stringify(canvasData, null, '\\t');\n}\n\nasync function applyFrontmatterChangesWithOffsets(\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(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 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 validateChanges(changes: FileChange[], content: string, frontmatter: CombinedFrontmatter<unknown>, path: string, shouldShowWarning: boolean): boolean {\n  const _debugger = getDebugger('validateChanges');\n  const logger = shouldShowWarning ? console.warn.bind(console) : _debugger;\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        logger('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        logger('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        logger('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        logger('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": []
}
