diff --git a/.gitignore b/.gitignore
index 3c3629e..cabc9a3 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1 +1,4 @@
node_modules
+main.js
+data.json
+package-lock.json
diff --git a/data.json b/data.json
deleted file mode 100644
index 9ba6454..0000000
--- a/data.json
+++ /dev/null
@@ -1,14 +0,0 @@
-{
- "micropubEndpoint": "https://blog.giersig.eu/micropub",
- "mediaEndpoint": "https://blog.giersig.eu/media",
- "accessToken": "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJtZSI6Imh0dHBzOi8vYmxvZy5naWVyc2lnLmV1Iiwic2NvcGUiOiJjcmVhdGUgdXBkYXRlIG1lZGlhIiwiaWF0IjoxNzczNTA1OTI2LCJleHAiOjE3ODEyODE5MjZ9.AaxtwgiQYbU7iWQt9GCResGK-DVfa0r7q2CHr_ibdFM",
- "defaultSyndicateTo": [],
- "autoDiscover": false,
- "siteUrl": "https://blog.giersig.eu",
- "authorizationEndpoint": "https://blog.giersig.eu/auth",
- "tokenEndpoint": "https://blog.giersig.eu/auth/token",
- "me": "https://blog.giersig.eu",
- "writeUrlToFrontmatter": true,
- "mapGardenTags": true,
- "defaultVisibility": "public"
-}
\ No newline at end of file
diff --git a/main.js b/main.js
deleted file mode 100644
index d66563d..0000000
--- a/main.js
+++ /dev/null
@@ -1,908 +0,0 @@
-/*
-THIS IS A GENERATED/BUNDLED FILE BY ESBUILD
-if you want to view the source, please visit the github repository of this plugin
-*/
-
-"use strict";
-var __create = Object.create;
-var __defProp = Object.defineProperty;
-var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
-var __getOwnPropNames = Object.getOwnPropertyNames;
-var __getProtoOf = Object.getPrototypeOf;
-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 __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
- // If the importer is in node compatibility mode or this is not an ESM
- // file that has been converted to a CommonJS file using a Babel-
- // compatible transform (i.e. "__esModule" has not been set), then set
- // "default" to the CommonJS "module.exports" for node compatibility.
- isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
- mod
-));
-var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
-
-// src/main.ts
-var main_exports = {};
-__export(main_exports, {
- default: () => MicropubPlugin
-});
-module.exports = __toCommonJS(main_exports);
-var import_obsidian5 = require("obsidian");
-
-// src/types.ts
-var DEFAULT_SETTINGS = {
- micropubEndpoint: "",
- mediaEndpoint: "",
- accessToken: "",
- defaultSyndicateTo: [],
- autoDiscover: false,
- siteUrl: "",
- authorizationEndpoint: "",
- tokenEndpoint: "",
- me: "",
- writeUrlToFrontmatter: true,
- mapGardenTags: true,
- defaultVisibility: "public"
-};
-
-// src/SettingsTab.ts
-var import_obsidian3 = require("obsidian");
-
-// src/MicropubClient.ts
-var import_obsidian = require("obsidian");
-var MicropubClient = class {
- constructor(getEndpoint, getMediaEndpoint, getToken) {
- this.getEndpoint = getEndpoint;
- this.getMediaEndpoint = getMediaEndpoint;
- this.getToken = getToken;
- }
- // ── Config discovery ─────────────────────────────────────────────────────
- /** Fetch Micropub server config (syndication targets, media endpoint, etc.) */
- async fetchConfig() {
- const url = `${this.getEndpoint()}?q=config`;
- const resp = await (0, import_obsidian.requestUrl)({
- url,
- method: "GET",
- headers: this.authHeaders()
- });
- return resp.json;
- }
- /**
- * Discover micropub + token endpoint URLs from a site's home page
- * by reading and tags.
- */
- async discoverEndpoints(siteUrl) {
- const resp = await (0, import_obsidian.requestUrl)({ url: siteUrl, method: "GET" });
- const html = resp.text;
- const micropub = this.extractLinkRel(html, "micropub");
- const tokenEndpoint = this.extractLinkRel(html, "token_endpoint");
- let mediaEndpoint;
- if (micropub) {
- try {
- const cfg = await this.fetchConfigFrom(micropub);
- mediaEndpoint = cfg["media-endpoint"];
- } catch (e) {
- }
- }
- return { micropubEndpoint: micropub, tokenEndpoint, mediaEndpoint };
- }
- // ── Post publishing ──────────────────────────────────────────────────────
- /**
- * Create a new post via Micropub.
- * Sends a JSON body with h-entry properties.
- * Returns the Location header URL on success.
- */
- async createPost(properties) {
- var _a, _b, _c;
- const body = {
- type: ["h-entry"],
- properties
- };
- try {
- const resp = await (0, import_obsidian.requestUrl)({
- url: this.getEndpoint(),
- method: "POST",
- headers: {
- ...this.authHeaders(),
- "Content-Type": "application/json"
- },
- body: JSON.stringify(body),
- throw: false
- });
- if (resp.status === 201 || resp.status === 202) {
- const location = ((_a = resp.headers) == null ? void 0 : _a["location"]) || ((_b = resp.headers) == null ? void 0 : _b["Location"]) || ((_c = resp.json) == null ? void 0 : _c.url);
- return { success: true, url: location };
- }
- const detail = this.extractError(resp.text);
- return { success: false, error: `HTTP ${resp.status}: ${detail}` };
- } catch (err) {
- return { success: false, error: String(err) };
- }
- }
- /**
- * Update an existing post.
- * @param postUrl The canonical URL of the post to update
- * @param replace Properties to replace (will overwrite existing values)
- */
- async updatePost(postUrl, replace) {
- const body = { action: "update", url: postUrl, replace };
- try {
- const resp = await (0, import_obsidian.requestUrl)({
- url: this.getEndpoint(),
- method: "POST",
- headers: {
- ...this.authHeaders(),
- "Content-Type": "application/json"
- },
- body: JSON.stringify(body),
- throw: false
- });
- if (resp.status >= 200 && resp.status < 300) {
- return { success: true, url: postUrl };
- }
- return {
- success: false,
- error: `HTTP ${resp.status}: ${this.extractError(resp.text)}`
- };
- } catch (err) {
- return { success: false, error: String(err) };
- }
- }
- // ── Media upload ─────────────────────────────────────────────────────────
- /**
- * Upload a binary file to the media endpoint.
- * @returns The URL of the uploaded media, or throws on failure.
- */
- async uploadMedia(fileBuffer, fileName, mimeType) {
- var _a, _b, _c;
- const endpoint = this.getMediaEndpoint() || `${this.getEndpoint()}/media`;
- const boundary = `----MicropubBoundary${Date.now()}`;
- const header = `--${boundary}\r
-Content-Disposition: form-data; name="file"; filename="${fileName}"\r
-Content-Type: ${mimeType}\r
-\r
-`;
- const footer = `\r
---${boundary}--\r
-`;
- const headerBuf = new TextEncoder().encode(header);
- const footerBuf = new TextEncoder().encode(footer);
- const fileBuf = new Uint8Array(fileBuffer);
- const combined = new Uint8Array(
- headerBuf.length + fileBuf.length + footerBuf.length
- );
- combined.set(headerBuf, 0);
- combined.set(fileBuf, headerBuf.length);
- combined.set(footerBuf, headerBuf.length + fileBuf.length);
- const resp = await (0, import_obsidian.requestUrl)({
- url: endpoint,
- method: "POST",
- headers: {
- ...this.authHeaders(),
- "Content-Type": `multipart/form-data; boundary=${boundary}`
- },
- body: combined.buffer,
- throw: false
- });
- if (resp.status === 201 || resp.status === 202) {
- const location = ((_a = resp.headers) == null ? void 0 : _a["location"]) || ((_b = resp.headers) == null ? void 0 : _b["Location"]) || ((_c = resp.json) == null ? void 0 : _c.url);
- if (location) return location;
- }
- throw new Error(
- `Media upload failed (HTTP ${resp.status}): ${this.extractError(resp.text)}`
- );
- }
- // ── Helpers ──────────────────────────────────────────────────────────────
- authHeaders() {
- return { Authorization: `Bearer ${this.getToken()}` };
- }
- extractLinkRel(html, rel) {
- var _a;
- const re = new RegExp(
- `]+rel=["']${rel}["'][^>]+href=["']([^"']+)["']|]+href=["']([^"']+)["'][^>]+rel=["']${rel}["']`,
- "i"
- );
- const m = html.match(re);
- return (_a = m == null ? void 0 : m[1]) != null ? _a : m == null ? void 0 : m[2];
- }
- async fetchConfigFrom(endpoint) {
- const resp = await (0, import_obsidian.requestUrl)({
- url: `${endpoint}?q=config`,
- method: "GET",
- headers: this.authHeaders()
- });
- return resp.json;
- }
- extractError(text) {
- var _a, _b;
- try {
- const obj = JSON.parse(text);
- return (_b = (_a = obj.error_description) != null ? _a : obj.error) != null ? _b : text.slice(0, 200);
- } catch (e) {
- return text.slice(0, 200);
- }
- }
-};
-
-// src/IndieAuth.ts
-var crypto = __toESM(require("crypto"));
-var import_obsidian2 = require("obsidian");
-var CLIENT_ID = "https://svemagie.github.io/obsidian-micropub/";
-var REDIRECT_URI = "https://svemagie.github.io/obsidian-micropub/callback";
-var SCOPE = "create update media";
-var AUTH_TIMEOUT_MS = 5 * 60 * 1e3;
-var pendingCallback = null;
-function handleProtocolCallback(params) {
- if (!pendingCallback) return;
- const { resolve, state: expectedState } = pendingCallback;
- pendingCallback = null;
- resolve(params);
-}
-var IndieAuth = class _IndieAuth {
- // ── Public API ────────────────────────────────────────────────────────────
- /**
- * Discover IndieAuth + Micropub endpoint URLs from the site's home page
- * by reading tags in the HTML
.
- */
- static async discoverEndpoints(siteUrl) {
- const resp = await (0, import_obsidian2.requestUrl)({ url: siteUrl, method: "GET" });
- const html = resp.text;
- const authorizationEndpoint = _IndieAuth.extractLinkRel(html, "authorization_endpoint");
- const tokenEndpoint = _IndieAuth.extractLinkRel(html, "token_endpoint");
- const micropubEndpoint = _IndieAuth.extractLinkRel(html, "micropub");
- if (!authorizationEndpoint) {
- throw new Error(
- `No found at ${siteUrl}. Make sure Indiekit is running and SITE_URL is set correctly.`
- );
- }
- if (!tokenEndpoint) {
- throw new Error(`No found at ${siteUrl}.`);
- }
- return { authorizationEndpoint, tokenEndpoint, micropubEndpoint };
- }
- /**
- * Run the full IndieAuth PKCE sign-in flow.
- *
- * Opens the browser at the user's IndieAuth login page. After login the
- * browser is redirected to the GitHub Pages callback, which triggers
- * the obsidian://micropub-auth protocol, which resolves the Promise here.
- *
- * Requires handleProtocolCallback() to be wired up in main.ts via
- * this.registerObsidianProtocolHandler("micropub-auth", handleProtocolCallback)
- */
- static async signIn(siteUrl) {
- var _a, _b, _c, _d, _e, _f;
- const { authorizationEndpoint, tokenEndpoint, micropubEndpoint } = await _IndieAuth.discoverEndpoints(siteUrl);
- const state = _IndieAuth.base64url(crypto.randomBytes(16));
- const codeVerifier = _IndieAuth.base64url(crypto.randomBytes(64));
- const codeChallenge = _IndieAuth.base64url(
- crypto.createHash("sha256").update(codeVerifier).digest()
- );
- const callbackPromise = new Promise(
- (resolve, reject) => {
- const timeout = setTimeout(() => {
- pendingCallback = null;
- reject(new Error("Sign-in timed out (5 min). Please try again."));
- }, AUTH_TIMEOUT_MS);
- pendingCallback = {
- state,
- resolve: (params) => {
- clearTimeout(timeout);
- resolve(params);
- }
- };
- }
- );
- const authUrl = new URL(authorizationEndpoint);
- authUrl.searchParams.set("response_type", "code");
- authUrl.searchParams.set("client_id", CLIENT_ID);
- authUrl.searchParams.set("redirect_uri", REDIRECT_URI);
- authUrl.searchParams.set("state", state);
- authUrl.searchParams.set("code_challenge", codeChallenge);
- authUrl.searchParams.set("code_challenge_method", "S256");
- authUrl.searchParams.set("scope", SCOPE);
- authUrl.searchParams.set("me", siteUrl);
- window.open(authUrl.toString());
- const callbackParams = await callbackPromise;
- if (callbackParams.state !== state) {
- throw new Error("State mismatch \u2014 possible CSRF attack. Please try again.");
- }
- const code = callbackParams.code;
- if (!code) {
- throw new Error(
- (_b = (_a = callbackParams.error_description) != null ? _a : callbackParams.error) != null ? _b : "No authorization code received."
- );
- }
- const tokenResp = await (0, import_obsidian2.requestUrl)({
- url: tokenEndpoint,
- method: "POST",
- headers: {
- "Content-Type": "application/x-www-form-urlencoded",
- Accept: "application/json"
- },
- body: new URLSearchParams({
- grant_type: "authorization_code",
- code,
- client_id: CLIENT_ID,
- redirect_uri: REDIRECT_URI,
- code_verifier: codeVerifier
- }).toString(),
- throw: false
- });
- const data = tokenResp.json;
- if (!data.access_token) {
- throw new Error(
- (_d = (_c = data.error_description) != null ? _c : data.error) != null ? _d : `Token exchange failed (HTTP ${tokenResp.status})`
- );
- }
- return {
- accessToken: data.access_token,
- scope: (_e = data.scope) != null ? _e : SCOPE,
- me: (_f = data.me) != null ? _f : siteUrl,
- authorizationEndpoint,
- tokenEndpoint,
- micropubEndpoint
- };
- }
- // ── Helpers ───────────────────────────────────────────────────────────────
- static base64url(buf) {
- return buf.toString("base64").replace(/\+/g, "-").replace(/\//g, "_").replace(/=/g, "");
- }
- static extractLinkRel(html, rel) {
- var _a;
- const re = new RegExp(
- `]+rel=["'][^"']*\\b${rel}\\b[^"']*["'][^>]+href=["']([^"']+)["']|]+href=["']([^"']+)["'][^>]+rel=["'][^"']*\\b${rel}\\b[^"']*["']`,
- "i"
- );
- const m = html.match(re);
- return (_a = m == null ? void 0 : m[1]) != null ? _a : m == null ? void 0 : m[2];
- }
-};
-
-// src/SettingsTab.ts
-var MicropubSettingsTab = class extends import_obsidian3.PluginSettingTab {
- constructor(app, plugin) {
- super(app, plugin);
- this.plugin = plugin;
- }
- display() {
- const { containerEl } = this;
- containerEl.empty();
- containerEl.createEl("h2", { text: "Micropub Publisher" });
- containerEl.createEl("h3", { text: "Account" });
- if (this.plugin.settings.me && this.plugin.settings.accessToken) {
- this.renderSignedIn(containerEl);
- } else {
- this.renderSignedOut(containerEl);
- }
- containerEl.createEl("h3", { text: "Endpoints" });
- containerEl.createEl("p", {
- text: "These are filled automatically when you sign in. Only edit them manually if your server uses non-standard paths.",
- cls: "setting-item-description"
- });
- new import_obsidian3.Setting(containerEl).setName("Micropub endpoint").setDesc("e.g. https://blog.giersig.eu/micropub").addText(
- (text) => text.setPlaceholder("https://example.com/micropub").setValue(this.plugin.settings.micropubEndpoint).onChange(async (value) => {
- this.plugin.settings.micropubEndpoint = value.trim();
- await this.plugin.saveSettings();
- })
- );
- new import_obsidian3.Setting(containerEl).setName("Media endpoint").setDesc("For image uploads. Auto-discovered if blank.").addText(
- (text) => text.setPlaceholder("https://example.com/micropub/media").setValue(this.plugin.settings.mediaEndpoint).onChange(async (value) => {
- this.plugin.settings.mediaEndpoint = value.trim();
- await this.plugin.saveSettings();
- })
- );
- containerEl.createEl("h3", { text: "Publish Behaviour" });
- new import_obsidian3.Setting(containerEl).setName("Default visibility").setDesc("Applies when the note has no explicit visibility property.").addDropdown(
- (drop) => drop.addOption("public", "Public").addOption("unlisted", "Unlisted").addOption("private", "Private").setValue(this.plugin.settings.defaultVisibility).onChange(async (value) => {
- this.plugin.settings.defaultVisibility = value;
- await this.plugin.saveSettings();
- })
- );
- new import_obsidian3.Setting(containerEl).setName("Write URL back to note").setDesc(
- "After publishing, store the post URL as `mp-url` in frontmatter. Subsequent publishes will update the existing post instead of creating a new one."
- ).addToggle(
- (toggle) => toggle.setValue(this.plugin.settings.writeUrlToFrontmatter).onChange(async (value) => {
- this.plugin.settings.writeUrlToFrontmatter = value;
- await this.plugin.saveSettings();
- })
- );
- containerEl.createEl("h3", { text: "Digital Garden" });
- new import_obsidian3.Setting(containerEl).setName("Map #garden/* tags to gardenStage").setDesc(
- "Obsidian tags like #garden/plant become a `garden-stage: plant` Micropub property. The blog renders these as growth stage badges at /garden/."
- ).addToggle(
- (toggle) => toggle.setValue(this.plugin.settings.mapGardenTags).onChange(async (value) => {
- this.plugin.settings.mapGardenTags = value;
- await this.plugin.saveSettings();
- })
- );
- containerEl.createEl("p", {
- text: "Stages: plant \u{1F331} \xB7 cultivate \u{1F33F} \xB7 question \u2753 \xB7 repot \u{1FAB4} \xB7 revitalize \u2728 \xB7 revisit \u{1F504}",
- cls: "setting-item-description"
- });
- }
- // ── Signed-out state ─────────────────────────────────────────────────────
- renderSignedOut(containerEl) {
- new import_obsidian3.Setting(containerEl).setName("Site URL").setDesc(
- "Your site's home page. Clicking Sign in opens your blog's login page in the browser \u2014 the same flow iA Writer uses."
- ).addText(
- (text) => text.setPlaceholder("https://blog.giersig.eu").setValue(this.plugin.settings.siteUrl).onChange(async (value) => {
- this.plugin.settings.siteUrl = value.trim();
- await this.plugin.saveSettings();
- })
- ).addButton((btn) => {
- btn.setButtonText("Sign in").setCta().onClick(async () => {
- const siteUrl = this.plugin.settings.siteUrl.trim();
- if (!siteUrl) {
- new import_obsidian3.Notice("Enter your site URL first.");
- return;
- }
- btn.setDisabled(true);
- btn.setButtonText("Opening browser\u2026");
- try {
- const result = await IndieAuth.signIn(siteUrl);
- this.plugin.settings.accessToken = result.accessToken;
- this.plugin.settings.me = result.me;
- this.plugin.settings.authorizationEndpoint = result.authorizationEndpoint;
- this.plugin.settings.tokenEndpoint = result.tokenEndpoint;
- if (result.micropubEndpoint) {
- this.plugin.settings.micropubEndpoint = result.micropubEndpoint;
- }
- if (result.mediaEndpoint) {
- this.plugin.settings.mediaEndpoint = result.mediaEndpoint;
- }
- await this.plugin.saveSettings();
- if (!this.plugin.settings.mediaEndpoint) {
- try {
- const client = new MicropubClient(
- () => this.plugin.settings.micropubEndpoint,
- () => this.plugin.settings.mediaEndpoint,
- () => this.plugin.settings.accessToken
- );
- const cfg = await client.fetchConfig();
- if (cfg["media-endpoint"]) {
- this.plugin.settings.mediaEndpoint = cfg["media-endpoint"];
- await this.plugin.saveSettings();
- }
- } catch (e) {
- }
- }
- new import_obsidian3.Notice(`\u2705 Signed in as ${result.me}`);
- this.display();
- } catch (err) {
- new import_obsidian3.Notice(`Sign-in failed: ${String(err)}`, 8e3);
- btn.setDisabled(false);
- btn.setButtonText("Sign in");
- }
- });
- });
- const details = containerEl.createEl("details");
- details.createEl("summary", {
- text: "Or paste a token manually",
- cls: "setting-item-description"
- });
- details.style.marginTop = "8px";
- details.style.marginBottom = "8px";
- new import_obsidian3.Setting(details).setName("Access token").setDesc("Bearer token from your Indiekit admin panel.").addText((text) => {
- text.setPlaceholder("your-bearer-token").setValue(this.plugin.settings.accessToken).onChange(async (value) => {
- this.plugin.settings.accessToken = value.trim();
- await this.plugin.saveSettings();
- });
- text.inputEl.type = "password";
- }).addButton(
- (btn) => btn.setButtonText("Verify").onClick(async () => {
- if (!this.plugin.settings.micropubEndpoint || !this.plugin.settings.accessToken) {
- new import_obsidian3.Notice("Set the Micropub endpoint and token first.");
- return;
- }
- btn.setDisabled(true);
- try {
- const client = new MicropubClient(
- () => this.plugin.settings.micropubEndpoint,
- () => this.plugin.settings.mediaEndpoint,
- () => this.plugin.settings.accessToken
- );
- await client.fetchConfig();
- new import_obsidian3.Notice("\u2705 Token is valid!");
- } catch (err) {
- new import_obsidian3.Notice(`Token check failed: ${String(err)}`);
- } finally {
- btn.setDisabled(false);
- }
- })
- );
- }
- // ── Signed-in state ──────────────────────────────────────────────────────
- renderSignedIn(containerEl) {
- const me = this.plugin.settings.me;
- const banner = containerEl.createDiv({
- cls: "micropub-auth-banner"
- });
- banner.style.cssText = "display:flex;align-items:center;gap:12px;padding:12px 16px;border:1px solid var(--background-modifier-border);border-radius:8px;margin-bottom:16px;background:var(--background-secondary);";
- const icon = banner.createDiv();
- icon.style.cssText = "width:40px;height:40px;border-radius:50%;background:var(--interactive-accent);display:flex;align-items:center;justify-content:center;font-size:1.2rem;flex-shrink:0;";
- icon.textContent = "\u{1F310}";
- const info = banner.createDiv();
- info.createEl("div", {
- text: "Signed in",
- attr: { style: "font-size:.75rem;color:var(--text-muted);margin-bottom:2px" }
- });
- info.createEl("div", {
- text: me,
- attr: { style: "font-weight:500;word-break:break-all" }
- });
- new import_obsidian3.Setting(containerEl).setName("Site URL").addText(
- (text) => text.setValue(this.plugin.settings.siteUrl).setDisabled(true)
- ).addButton(
- (btn) => btn.setButtonText("Sign out").setWarning().onClick(async () => {
- this.plugin.settings.accessToken = "";
- this.plugin.settings.me = "";
- this.plugin.settings.authorizationEndpoint = "";
- this.plugin.settings.tokenEndpoint = "";
- await this.plugin.saveSettings();
- this.display();
- })
- );
- }
-};
-
-// src/Publisher.ts
-var import_obsidian4 = require("obsidian");
-var GARDEN_TAG_PREFIX = "garden/";
-var Publisher = class {
- constructor(app, settings) {
- this.app = app;
- this.settings = settings;
- this.client = new MicropubClient(
- () => settings.micropubEndpoint,
- () => settings.mediaEndpoint,
- () => settings.accessToken
- );
- }
- /** Publish the given file. Returns PublishResult. */
- async publish(file) {
- var _a, _b;
- const raw = await this.app.vault.read(file);
- const { frontmatter, body } = this.parseFrontmatter(raw);
- const existingUrl = (_b = (_a = frontmatter["mp-url"]) != null ? _a : frontmatter["url"]) != null ? _b : void 0;
- const { content: processedBody, uploadedUrls } = await this.processImages(body);
- const properties = this.buildProperties(frontmatter, processedBody, uploadedUrls);
- let result;
- if (existingUrl) {
- const replace = {};
- for (const [k, v] of Object.entries(properties)) {
- replace[k] = Array.isArray(v) ? v : [v];
- }
- result = await this.client.updatePost(existingUrl, replace);
- } else {
- result = await this.client.createPost(properties);
- }
- if (result.success && result.url && this.settings.writeUrlToFrontmatter) {
- await this.writeUrlToNote(file, raw, result.url);
- }
- return result;
- }
- // ── Property builder ─────────────────────────────────────────────────────
- buildProperties(fm, body, uploadedUrls) {
- var _a, _b, _c, _d, _e, _f, _g, _h, _i, _j;
- const props = {};
- const trimmedBody = body.trim();
- const bookmarkOf = (_a = fm["bookmarkOf"]) != null ? _a : fm["bookmark-of"];
- const likeOf = (_b = fm["likeOf"]) != null ? _b : fm["like-of"];
- const inReplyTo = (_c = fm["inReplyTo"]) != null ? _c : fm["in-reply-to"];
- const repostOf = (_d = fm["repostOf"]) != null ? _d : fm["repost-of"];
- if (bookmarkOf) props["bookmark-of"] = [String(bookmarkOf)];
- if (likeOf) props["like-of"] = [String(likeOf)];
- if (inReplyTo) props["in-reply-to"] = [String(inReplyTo)];
- if (repostOf) props["repost-of"] = [String(repostOf)];
- const isInteractionWithoutBody = (likeOf || repostOf) && !trimmedBody;
- if (!isInteractionWithoutBody) {
- props["content"] = trimmedBody ? [{ html: trimmedBody }] : [{ html: "" }];
- }
- if (fm["title"]) {
- props["name"] = [String(fm["title"])];
- }
- if ((_e = fm["summary"]) != null ? _e : fm["excerpt"]) {
- props["summary"] = [String((_f = fm["summary"]) != null ? _f : fm["excerpt"])];
- }
- const rawDate = (_g = fm["created"]) != null ? _g : fm["date"];
- if (rawDate) {
- props["published"] = [new Date(String(rawDate)).toISOString()];
- }
- const rawTags = this.resolveArray((_h = fm["tags"]) != null ? _h : fm["category"]);
- const gardenStage = this.extractGardenStage(rawTags);
- const normalTags = rawTags.filter(
- (t) => !t.startsWith(GARDEN_TAG_PREFIX) && t !== "garden"
- );
- if (normalTags.length > 0) {
- props["category"] = normalTags;
- }
- if (this.settings.mapGardenTags && gardenStage) {
- props["garden-stage"] = [gardenStage];
- }
- const syndicateTo = this.resolveArray(
- (_i = fm["mp-syndicate-to"]) != null ? _i : fm["mpSyndicateTo"]
- );
- const allSyndicateTo = [
- .../* @__PURE__ */ new Set([...this.settings.defaultSyndicateTo, ...syndicateTo])
- ];
- if (allSyndicateTo.length > 0) {
- props["mp-syndicate-to"] = allSyndicateTo;
- }
- const visibility = (_j = fm["visibility"]) != null ? _j : this.settings.defaultVisibility;
- if (visibility && visibility !== "public") {
- props["visibility"] = [visibility];
- }
- if (fm["ai"] && typeof fm["ai"] === "object") {
- props["ai"] = [fm["ai"]];
- }
- const fmPhotos = this.resolvePhotoArray(fm["photo"]);
- if (fmPhotos.length > 0) {
- props["photo"] = fmPhotos;
- } else if (uploadedUrls.length > 0) {
- props["photo"] = uploadedUrls.map((url) => ({ value: url }));
- }
- for (const [k, v] of Object.entries(fm)) {
- if (k.startsWith("mp-") && k !== "mp-url" && k !== "mp-syndicate-to") {
- props[k] = this.resolveArray(v);
- }
- }
- return props;
- }
- /**
- * Normalise the `photo` frontmatter field into Micropub photo objects.
- * Handles three formats:
- * - string URL: "https://..."
- * - array of strings: ["https://..."]
- * - array of objects: [{url: "https://...", alt: "..."}]
- */
- resolvePhotoArray(value) {
- if (!value) return [];
- const items = Array.isArray(value) ? value : [value];
- return items.map((item) => {
- var _a, _b;
- if (typeof item === "string") return { value: item };
- if (typeof item === "object" && item !== null) {
- const obj = item;
- const url = String((_b = (_a = obj["url"]) != null ? _a : obj["value"]) != null ? _b : "");
- if (!url) return null;
- return obj["alt"] ? { value: url, alt: String(obj["alt"]) } : { value: url };
- }
- return null;
- }).filter((x) => x !== null);
- }
- // ── Garden tag extraction ────────────────────────────────────────────────
- /**
- * Find the first #garden/ tag and return the stage name.
- * Supports both "garden/plant" (Obsidian array) and "#garden/plant" (inline).
- */
- extractGardenStage(tags) {
- for (const tag of tags) {
- const clean = tag.replace(/^#/, "");
- if (clean.startsWith(GARDEN_TAG_PREFIX)) {
- const stage = clean.slice(GARDEN_TAG_PREFIX.length);
- const valid = [
- "plant",
- "cultivate",
- "question",
- "repot",
- "revitalize",
- "revisit"
- ];
- if (valid.includes(stage)) return stage;
- }
- }
- return void 0;
- }
- // ── Image processing ─────────────────────────────────────────────────────
- /**
- * Find all `![[local-image.png]]` or `` in the body,
- * upload them to the media endpoint, and replace the references with remote URLs.
- */
- async processImages(body) {
- const uploadedUrls = [];
- const wikiPattern = /!\[\[([^\]]+\.(png|jpg|jpeg|gif|webp|svg))\]\]/gi;
- const mdPattern = /!\[([^\]]*)\]\(([^)]+\.(png|jpg|jpeg|gif|webp|svg))\)/gi;
- let content = body;
- const wikiMatches = [...body.matchAll(wikiPattern)];
- for (const match of wikiMatches) {
- const filename = match[1];
- try {
- const remoteUrl = await this.uploadLocalFile(filename);
- if (remoteUrl) {
- uploadedUrls.push(remoteUrl);
- content = content.replace(match[0], ``);
- }
- } catch (err) {
- console.warn(`[micropub] Failed to upload ${filename}:`, err);
- }
- }
- const mdMatches = [...content.matchAll(mdPattern)];
- for (const match of mdMatches) {
- const alt = match[1];
- const path = match[2];
- if (path.startsWith("http")) continue;
- try {
- const remoteUrl = await this.uploadLocalFile(path);
- if (remoteUrl) {
- uploadedUrls.push(remoteUrl);
- content = content.replace(match[0], ``);
- }
- } catch (err) {
- console.warn(`[micropub] Failed to upload ${path}:`, err);
- }
- }
- return { content, uploadedUrls };
- }
- async uploadLocalFile(path) {
- const file = this.app.vault.getFiles().find(
- (f) => f.name === path || f.path === path
- );
- if (!file) return void 0;
- const buffer = await this.app.vault.readBinary(file);
- const mimeType = this.guessMimeType(file.extension);
- return this.client.uploadMedia(buffer, file.name, mimeType);
- }
- // ── Frontmatter helpers ──────────────────────────────────────────────────
- parseFrontmatter(raw) {
- var _a;
- const fmMatch = raw.match(/^---\r?\n([\s\S]*?)\r?\n---\r?\n([\s\S]*)$/);
- if (!fmMatch) return { frontmatter: {}, body: raw };
- let frontmatter = {};
- try {
- frontmatter = (_a = (0, import_obsidian4.parseYaml)(fmMatch[1])) != null ? _a : {};
- } catch (e) {
- }
- return { frontmatter, body: fmMatch[2] };
- }
- async writeUrlToNote(file, originalContent, url) {
- const fmMatch = originalContent.match(
- /^(---\r?\n[\s\S]*?\r?\n---\r?\n)([\s\S]*)$/
- );
- if (!fmMatch) {
- const newFm = `---
-mp-url: "${url}"
----
-`;
- await this.app.vault.modify(file, newFm + originalContent);
- return;
- }
- const fmBlock = fmMatch[1];
- const body = fmMatch[2];
- if (fmBlock.includes("mp-url:")) {
- const updated = fmBlock.replace(
- /mp-url:.*(\r?\n)/,
- `mp-url: "${url}"$1`
- );
- await this.app.vault.modify(file, updated + body);
- } else {
- const updated = fmBlock.replace(
- /(\r?\n---\r?\n)$/,
- `
-mp-url: "${url}"$1`
- );
- await this.app.vault.modify(file, updated + body);
- }
- }
- resolveArray(value) {
- if (!value) return [];
- if (Array.isArray(value)) return value.map(String);
- return [String(value)];
- }
- guessMimeType(ext) {
- var _a;
- const map = {
- png: "image/png",
- jpg: "image/jpeg",
- jpeg: "image/jpeg",
- gif: "image/gif",
- webp: "image/webp",
- svg: "image/svg+xml"
- };
- return (_a = map[ext.toLowerCase()]) != null ? _a : "application/octet-stream";
- }
-};
-
-// src/main.ts
-var MicropubPlugin = class extends import_obsidian5.Plugin {
- async onload() {
- await this.loadSettings();
- this.addCommand({
- id: "publish-to-micropub",
- name: "Publish to Micropub",
- checkCallback: (checking) => {
- const file = this.app.workspace.getActiveFile();
- if (!file || file.extension !== "md") return false;
- if (checking) return true;
- this.publishActiveNote(file);
- return true;
- }
- });
- this.addCommand({
- id: "publish-to-micropub-update",
- name: "Update existing Micropub post",
- checkCallback: (checking) => {
- const file = this.app.workspace.getActiveFile();
- if (!file || file.extension !== "md") return false;
- if (checking) return true;
- this.publishActiveNote(file);
- return true;
- }
- });
- this.registerObsidianProtocolHandler("micropub-auth", (params) => {
- handleProtocolCallback(params);
- });
- this.addSettingTab(new MicropubSettingsTab(this.app, this));
- this.addRibbonIcon("send", "Publish to Micropub", () => {
- const file = this.app.workspace.getActiveFile();
- if (!file || file.extension !== "md") {
- new import_obsidian5.Notice("Open a Markdown note to publish.");
- return;
- }
- this.publishActiveNote(file);
- });
- }
- onunload() {
- }
- // ── Publish flow ──────────────────────────────────────────────────────────
- async publishActiveNote(file) {
- if (!this.settings.micropubEndpoint) {
- new import_obsidian5.Notice(
- "\u26A0\uFE0F Micropub endpoint not configured. Open plugin settings to add it."
- );
- return;
- }
- if (!this.settings.accessToken) {
- new import_obsidian5.Notice(
- "\u26A0\uFE0F Access token not configured. Open plugin settings to add it."
- );
- return;
- }
- const notice = new import_obsidian5.Notice(
- "Publishing\u2026",
- 0
- /* persist until dismissed */
- );
- try {
- const publisher = new Publisher(this.app, this.settings);
- const result = await publisher.publish(file);
- notice.hide();
- if (result.success) {
- const urlDisplay = result.url ? `
-${result.url}` : "";
- new import_obsidian5.Notice(`\u2705 Published!${urlDisplay}`, 8e3);
- } else {
- new import_obsidian5.Notice(`\u274C Publish failed: ${result.error}`, 1e4);
- console.error("[micropub] Publish failed:", result.error);
- }
- } catch (err) {
- notice.hide();
- const msg = err instanceof Error ? err.message : String(err);
- new import_obsidian5.Notice(`\u274C Error: ${msg}`, 1e4);
- console.error("[micropub] Unexpected error:", err);
- }
- }
- // ── Settings persistence ──────────────────────────────────────────────────
- async loadSettings() {
- this.settings = Object.assign(
- {},
- DEFAULT_SETTINGS,
- await this.loadData()
- );
- }
- async saveSettings() {
- await this.saveData(this.settings);
- }
-};
-//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/main.ts", "src/types.ts", "src/SettingsTab.ts", "src/MicropubClient.ts", "src/IndieAuth.ts", "src/Publisher.ts"],
  "sourcesContent": ["/**\n * main.ts \u2014 obsidian-micropub plugin entry point\n *\n * Publishes the active note to any Micropub-compatible endpoint.\n * Designed to work with Indiekit (https://getindiekit.com) but compatible\n * with any server that implements the Micropub spec (W3C).\n *\n * Key features vs. the original obsidian-microblog:\n *   - Configurable endpoint URL (not hardcoded to Micro.blog)\n *   - Auto-discovery of micropub/media endpoints from <link rel> headers\n *   - #garden/* tag \u2192 gardenStage property mapping for Digital Garden\n *   - Writes returned post URL back to note frontmatter for future updates\n *   - Supports create + update flows\n *\n * Based on: https://github.com/svemagie/obsidian-microblog (MIT)\n */\n\nimport { Notice, Plugin, TFile } from \"obsidian\";\nimport { DEFAULT_SETTINGS, type MicropubSettings } from \"./types\";\nimport { MicropubSettingsTab } from \"./SettingsTab\";\nimport { Publisher } from \"./Publisher\";\nimport { handleProtocolCallback } from \"./IndieAuth\";\n\nexport default class MicropubPlugin extends Plugin {\n  settings!: MicropubSettings;\n\n  async onload(): Promise<void> {\n    await this.loadSettings();\n\n    // \u2500\u2500 Commands \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n    this.addCommand({\n      id: \"publish-to-micropub\",\n      name: \"Publish to Micropub\",\n      checkCallback: (checking: boolean) => {\n        const file = this.app.workspace.getActiveFile();\n        if (!file || file.extension !== \"md\") return false;\n        if (checking) return true;\n\n        this.publishActiveNote(file);\n        return true;\n      },\n    });\n\n    this.addCommand({\n      id: \"publish-to-micropub-update\",\n      name: \"Update existing Micropub post\",\n      checkCallback: (checking: boolean) => {\n        const file = this.app.workspace.getActiveFile();\n        if (!file || file.extension !== \"md\") return false;\n        if (checking) return true;\n\n        // Update uses the same publish flow \u2014 Publisher detects mp-url and routes to update\n        this.publishActiveNote(file);\n        return true;\n      },\n    });\n\n    // \u2500\u2500 IndieAuth protocol handler \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    // Receives obsidian://micropub-auth?code=...&state=... after the user\n    // approves on their IndieAuth login page. The GitHub Pages callback page\n    // at svemagie.github.io/obsidian-micropub/callback redirects here.\n    this.registerObsidianProtocolHandler(\"micropub-auth\", (params) => {\n      handleProtocolCallback(params as Record<string, string>);\n    });\n\n    // \u2500\u2500 Settings tab \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n    this.addSettingTab(new MicropubSettingsTab(this.app, this));\n\n    // \u2500\u2500 Ribbon icon \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n    this.addRibbonIcon(\"send\", \"Publish to Micropub\", () => {\n      const file = this.app.workspace.getActiveFile();\n      if (!file || file.extension !== \"md\") {\n        new Notice(\"Open a Markdown note to publish.\");\n        return;\n      }\n      this.publishActiveNote(file);\n    });\n  }\n\n  onunload(): void {\n    // Nothing to clean up\n  }\n\n  // \u2500\u2500 Publish flow \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  private async publishActiveNote(file: TFile): Promise<void> {\n    if (!this.settings.micropubEndpoint) {\n      new Notice(\n        \"\u26A0\uFE0F Micropub endpoint not configured. Open plugin settings to add it.\",\n      );\n      return;\n    }\n\n    if (!this.settings.accessToken) {\n      new Notice(\n        \"\u26A0\uFE0F Access token not configured. Open plugin settings to add it.\",\n      );\n      return;\n    }\n\n    const notice = new Notice(\"Publishing\u2026\", 0 /* persist until dismissed */);\n\n    try {\n      const publisher = new Publisher(this.app, this.settings);\n      const result = await publisher.publish(file);\n\n      notice.hide();\n\n      if (result.success) {\n        const urlDisplay = result.url\n          ? `\\n${result.url}`\n          : \"\";\n        new Notice(`\u2705 Published!${urlDisplay}`, 8000);\n      } else {\n        new Notice(`\u274C Publish failed: ${result.error}`, 10000);\n        console.error(\"[micropub] Publish failed:\", result.error);\n      }\n    } catch (err: unknown) {\n      notice.hide();\n      const msg = err instanceof Error ? err.message : String(err);\n      new Notice(`\u274C Error: ${msg}`, 10000);\n      console.error(\"[micropub] Unexpected error:\", err);\n    }\n  }\n\n  // \u2500\u2500 Settings persistence \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  async loadSettings(): Promise<void> {\n    this.settings = Object.assign(\n      {},\n      DEFAULT_SETTINGS,\n      await this.loadData(),\n    ) as MicropubSettings;\n  }\n\n  async saveSettings(): Promise<void> {\n    await this.saveData(this.settings);\n  }\n}\n", "/**\n * types.ts \u2014 shared interfaces for obsidian-micropub\n */\n\n/** Plugin settings stored in data.json */\nexport interface MicropubSettings {\n  /** Full URL of the Micropub endpoint, e.g. https://example.com/micropub */\n  micropubEndpoint: string;\n\n  /**\n   * Full URL of the media endpoint for image uploads.\n   * If empty, discovered automatically from the Micropub config query,\n   * or derived from the micropubEndpoint (some servers use /micropub/media).\n   */\n  mediaEndpoint: string;\n\n  /**\n   * Bearer token for Authorization: Bearer <token>.\n   * Obtain from your IndieAuth token endpoint or server admin panel.\n   */\n  accessToken: string;\n\n  /**\n   * The syndication targets to pre-tick in the publish dialog.\n   * Values are uid strings returned by the Micropub config ?q=config.\n   */\n  defaultSyndicateTo: string[];\n\n  /**\n   * When true, perform a discovery fetch against the site URL to auto-detect\n   * the micropub and token endpoints from <link rel=\"micropub\"> headers.\n   */\n  autoDiscover: boolean;\n\n  /** Your site's homepage URL \u2014 used for endpoint discovery and IndieAuth. */\n  siteUrl: string;\n\n  /**\n   * The authorization_endpoint discovered from the site.\n   * Populated automatically by the IndieAuth sign-in flow.\n   */\n  authorizationEndpoint: string;\n\n  /**\n   * The token_endpoint discovered from the site.\n   * Populated automatically by the IndieAuth sign-in flow.\n   */\n  tokenEndpoint: string;\n\n  /**\n   * The canonical \"me\" URL returned by the token endpoint after sign-in.\n   * Used to show who is currently logged in.\n   */\n  me: string;\n\n  /**\n   * When true, after a successful publish the post URL returned by the server\n   * is written back to the note's frontmatter as `mp-url`.\n   */\n  writeUrlToFrontmatter: boolean;\n\n  /**\n   * Map Obsidian #garden/* tags to a `gardenStage` Micropub property.\n   * When enabled, a tag like #garden/plant becomes { \"garden-stage\": \"plant\" }\n   * in the Micropub request (and gardenStage: plant in the server's front matter).\n   */\n  mapGardenTags: boolean;\n\n  /** Visibility default for new posts: \"public\" | \"unlisted\" | \"private\" */\n  defaultVisibility: \"public\" | \"unlisted\" | \"private\";\n}\n\nexport const DEFAULT_SETTINGS: MicropubSettings = {\n  micropubEndpoint: \"\",\n  mediaEndpoint: \"\",\n  accessToken: \"\",\n  defaultSyndicateTo: [],\n  autoDiscover: false,\n  siteUrl: \"\",\n  authorizationEndpoint: \"\",\n  tokenEndpoint: \"\",\n  me: \"\",\n  writeUrlToFrontmatter: true,\n  mapGardenTags: true,\n  defaultVisibility: \"public\",\n};\n\n/** A syndication target as returned by Micropub config query */\nexport interface SyndicationTarget {\n  uid: string;\n  name: string;\n}\n\n/** Micropub config response (?q=config) */\nexport interface MicropubConfig {\n  \"media-endpoint\"?: string;\n  \"syndicate-to\"?: SyndicationTarget[];\n  \"post-types\"?: Array<{ type: string; name: string }>;\n}\n\n/**\n * Garden stages \u2014 matches Obsidian #garden/* tags and blog gardenStage values.\n * The Micropub property name is \"garden-stage\" (hyphenated, Micropub convention).\n */\nexport type GardenStage =\n  | \"plant\"\n  | \"cultivate\"\n  | \"question\"\n  | \"repot\"\n  | \"revitalize\"\n  | \"revisit\";\n\nexport const GARDEN_STAGE_LABELS: Record<GardenStage, string> = {\n  plant:      \"\uD83C\uDF31 Seedling\",\n  cultivate:  \"\uD83C\uDF3F Growing\",\n  question:   \"\u2753 Open Question\",\n  repot:      \"\uD83E\uDEB4 Repotting\",\n  revitalize: \"\u2728 Revitalizing\",\n  revisit:    \"\uD83D\uDD04 Revisit\",\n};\n\n/** Result returned by Publisher.publish() */\nexport interface PublishResult {\n  success: boolean;\n  /** URL of the published post (from Location response header) */\n  url?: string;\n  error?: string;\n}\n", "/**\n * SettingsTab.ts \u2014 Obsidian settings UI for obsidian-micropub\n *\n * Authentication section works like iA Writer:\n *   1. User enters their site URL\n *   2. Clicks \"Sign in\" \u2014 browser opens at their IndieAuth login page\n *   3. They log in with their blog password\n *   4. Browser redirects back; plugin receives the token automatically\n *   5. Settings show \"Signed in as <me>\" + a Sign Out button\n *\n * Advanced users can still paste a token manually if they prefer.\n */\n\nimport { App, Notice, PluginSettingTab, Setting } from \"obsidian\";\nimport type MicropubPlugin from \"./main\";\nimport { MicropubClient } from \"./MicropubClient\";\nimport { IndieAuth } from \"./IndieAuth\";\n\nexport class MicropubSettingsTab extends PluginSettingTab {\n  constructor(\n    app: App,\n    private readonly plugin: MicropubPlugin,\n  ) {\n    super(app, plugin);\n  }\n\n  display(): void {\n    const { containerEl } = this;\n    containerEl.empty();\n\n    containerEl.createEl(\"h2\", { text: \"Micropub Publisher\" });\n\n    // \u2500\u2500 Site URL + Sign In \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    containerEl.createEl(\"h3\", { text: \"Account\" });\n\n    // Show current sign-in status\n    if (this.plugin.settings.me && this.plugin.settings.accessToken) {\n      this.renderSignedIn(containerEl);\n    } else {\n      this.renderSignedOut(containerEl);\n    }\n\n    // \u2500\u2500 Endpoints (collapsed / advanced) \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    containerEl.createEl(\"h3\", { text: \"Endpoints\" });\n\n    containerEl.createEl(\"p\", {\n      text: \"These are filled automatically when you sign in. Only edit them manually if your server uses non-standard paths.\",\n      cls: \"setting-item-description\",\n    });\n\n    new Setting(containerEl)\n      .setName(\"Micropub endpoint\")\n      .setDesc(\"e.g. https://blog.giersig.eu/micropub\")\n      .addText((text) =>\n        text\n          .setPlaceholder(\"https://example.com/micropub\")\n          .setValue(this.plugin.settings.micropubEndpoint)\n          .onChange(async (value) => {\n            this.plugin.settings.micropubEndpoint = value.trim();\n            await this.plugin.saveSettings();\n          }),\n      );\n\n    new Setting(containerEl)\n      .setName(\"Media endpoint\")\n      .setDesc(\"For image uploads. Auto-discovered if blank.\")\n      .addText((text) =>\n        text\n          .setPlaceholder(\"https://example.com/micropub/media\")\n          .setValue(this.plugin.settings.mediaEndpoint)\n          .onChange(async (value) => {\n            this.plugin.settings.mediaEndpoint = value.trim();\n            await this.plugin.saveSettings();\n          }),\n      );\n\n    // \u2500\u2500 Publish behaviour \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    containerEl.createEl(\"h3\", { text: \"Publish Behaviour\" });\n\n    new Setting(containerEl)\n      .setName(\"Default visibility\")\n      .setDesc(\"Applies when the note has no explicit visibility property.\")\n      .addDropdown((drop) =>\n        drop\n          .addOption(\"public\", \"Public\")\n          .addOption(\"unlisted\", \"Unlisted\")\n          .addOption(\"private\", \"Private\")\n          .setValue(this.plugin.settings.defaultVisibility)\n          .onChange(async (value) => {\n            this.plugin.settings.defaultVisibility = value as\n              | \"public\"\n              | \"unlisted\"\n              | \"private\";\n            await this.plugin.saveSettings();\n          }),\n      );\n\n    new Setting(containerEl)\n      .setName(\"Write URL back to note\")\n      .setDesc(\n        \"After publishing, store the post URL as `mp-url` in frontmatter. \" +\n        \"Subsequent publishes will update the existing post instead of creating a new one.\",\n      )\n      .addToggle((toggle) =>\n        toggle\n          .setValue(this.plugin.settings.writeUrlToFrontmatter)\n          .onChange(async (value) => {\n            this.plugin.settings.writeUrlToFrontmatter = value;\n            await this.plugin.saveSettings();\n          }),\n      );\n\n    // \u2500\u2500 Digital Garden \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    containerEl.createEl(\"h3\", { text: \"Digital Garden\" });\n\n    new Setting(containerEl)\n      .setName(\"Map #garden/* tags to gardenStage\")\n      .setDesc(\n        \"Obsidian tags like #garden/plant become a `garden-stage: plant` Micropub \" +\n        \"property. The blog renders these as growth stage badges at /garden/.\",\n      )\n      .addToggle((toggle) =>\n        toggle\n          .setValue(this.plugin.settings.mapGardenTags)\n          .onChange(async (value) => {\n            this.plugin.settings.mapGardenTags = value;\n            await this.plugin.saveSettings();\n          }),\n      );\n\n    containerEl.createEl(\"p\", {\n      text: \"Stages: plant \uD83C\uDF31 \u00B7 cultivate \uD83C\uDF3F \u00B7 question \u2753 \u00B7 repot \uD83E\uDEB4 \u00B7 revitalize \u2728 \u00B7 revisit \uD83D\uDD04\",\n      cls: \"setting-item-description\",\n    });\n  }\n\n  // \u2500\u2500 Signed-out state \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  private renderSignedOut(containerEl: HTMLElement): void {\n    // Site URL input + Sign In button on the same row\n    new Setting(containerEl)\n      .setName(\"Site URL\")\n      .setDesc(\n        \"Your site's home page. Clicking Sign in opens your blog's login page \" +\n        \"in the browser \u2014 the same flow iA Writer uses.\",\n      )\n      .addText((text) =>\n        text\n          .setPlaceholder(\"https://blog.giersig.eu\")\n          .setValue(this.plugin.settings.siteUrl)\n          .onChange(async (value) => {\n            this.plugin.settings.siteUrl = value.trim();\n            await this.plugin.saveSettings();\n          }),\n      )\n      .addButton((btn) => {\n        btn\n          .setButtonText(\"Sign in\")\n          .setCta()\n          .onClick(async () => {\n            const siteUrl = this.plugin.settings.siteUrl.trim();\n            if (!siteUrl) {\n              new Notice(\"Enter your site URL first.\");\n              return;\n            }\n\n            btn.setDisabled(true);\n            btn.setButtonText(\"Opening browser\u2026\");\n\n            try {\n              const result = await IndieAuth.signIn(siteUrl);\n\n              // Save everything returned by the auth flow\n              this.plugin.settings.accessToken      = result.accessToken;\n              this.plugin.settings.me               = result.me;\n              this.plugin.settings.authorizationEndpoint = result.authorizationEndpoint;\n              this.plugin.settings.tokenEndpoint    = result.tokenEndpoint;\n              if (result.micropubEndpoint) {\n                this.plugin.settings.micropubEndpoint = result.micropubEndpoint;\n              }\n              if (result.mediaEndpoint) {\n                this.plugin.settings.mediaEndpoint = result.mediaEndpoint;\n              }\n\n              await this.plugin.saveSettings();\n\n              // Try to fetch the Micropub config to pick up media endpoint\n              if (!this.plugin.settings.mediaEndpoint) {\n                try {\n                  const client = new MicropubClient(\n                    () => this.plugin.settings.micropubEndpoint,\n                    () => this.plugin.settings.mediaEndpoint,\n                    () => this.plugin.settings.accessToken,\n                  );\n                  const cfg = await client.fetchConfig();\n                  if (cfg[\"media-endpoint\"]) {\n                    this.plugin.settings.mediaEndpoint = cfg[\"media-endpoint\"];\n                    await this.plugin.saveSettings();\n                  }\n                } catch {\n                  // Non-fatal\n                }\n              }\n\n              new Notice(`\u2705 Signed in as ${result.me}`);\n              this.display(); // Refresh to show signed-in state\n            } catch (err: unknown) {\n              new Notice(`Sign-in failed: ${String(err)}`, 8000);\n              btn.setDisabled(false);\n              btn.setButtonText(\"Sign in\");\n            }\n          });\n      });\n\n    // Divider + manual token fallback (collapsed by default)\n    const details = containerEl.createEl(\"details\");\n    details.createEl(\"summary\", {\n      text: \"Or paste a token manually\",\n      cls: \"setting-item-description\",\n    });\n    details.style.marginTop = \"8px\";\n    details.style.marginBottom = \"8px\";\n\n    new Setting(details)\n      .setName(\"Access token\")\n      .setDesc(\"Bearer token from your Indiekit admin panel.\")\n      .addText((text) => {\n        text\n          .setPlaceholder(\"your-bearer-token\")\n          .setValue(this.plugin.settings.accessToken)\n          .onChange(async (value) => {\n            this.plugin.settings.accessToken = value.trim();\n            await this.plugin.saveSettings();\n          });\n        text.inputEl.type = \"password\";\n      })\n      .addButton((btn) =>\n        btn.setButtonText(\"Verify\").onClick(async () => {\n          if (\n            !this.plugin.settings.micropubEndpoint ||\n            !this.plugin.settings.accessToken\n          ) {\n            new Notice(\"Set the Micropub endpoint and token first.\");\n            return;\n          }\n          btn.setDisabled(true);\n          try {\n            const client = new MicropubClient(\n              () => this.plugin.settings.micropubEndpoint,\n              () => this.plugin.settings.mediaEndpoint,\n              () => this.plugin.settings.accessToken,\n            );\n            await client.fetchConfig();\n            new Notice(\"\u2705 Token is valid!\");\n          } catch (err: unknown) {\n            new Notice(`Token check failed: ${String(err)}`);\n          } finally {\n            btn.setDisabled(false);\n          }\n        }),\n      );\n  }\n\n  // \u2500\u2500 Signed-in state \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  private renderSignedIn(containerEl: HTMLElement): void {\n    const me = this.plugin.settings.me;\n\n    // Avatar + \"Signed in as\" banner\n    const banner = containerEl.createDiv({\n      cls: \"micropub-auth-banner\",\n    });\n    banner.style.cssText =\n      \"display:flex;align-items:center;gap:12px;padding:12px 16px;\" +\n      \"border:1px solid var(--background-modifier-border);\" +\n      \"border-radius:8px;margin-bottom:16px;background:var(--background-secondary);\";\n\n    const icon = banner.createDiv();\n    icon.style.cssText =\n      \"width:40px;height:40px;border-radius:50%;background:var(--interactive-accent);\" +\n      \"display:flex;align-items:center;justify-content:center;\" +\n      \"font-size:1.2rem;flex-shrink:0;\";\n    icon.textContent = \"\uD83C\uDF10\";\n\n    const info = banner.createDiv();\n    info.createEl(\"div\", {\n      text: \"Signed in\",\n      attr: { style: \"font-size:.75rem;color:var(--text-muted);margin-bottom:2px\" },\n    });\n    info.createEl(\"div\", {\n      text: me,\n      attr: { style: \"font-weight:500;word-break:break-all\" },\n    });\n\n    new Setting(containerEl)\n      .setName(\"Site URL\")\n      .addText((text) =>\n        text\n          .setValue(this.plugin.settings.siteUrl)\n          .setDisabled(true),\n      )\n      .addButton((btn) =>\n        btn\n          .setButtonText(\"Sign out\")\n          .setWarning()\n          .onClick(async () => {\n            this.plugin.settings.accessToken = \"\";\n            this.plugin.settings.me = \"\";\n            this.plugin.settings.authorizationEndpoint = \"\";\n            this.plugin.settings.tokenEndpoint = \"\";\n            await this.plugin.saveSettings();\n            this.display();\n          }),\n      );\n  }\n}\n", "/**\n * MicropubClient.ts\n *\n * Low-level HTTP client for Micropub and Media endpoint requests.\n * Uses Obsidian's requestUrl() so requests are made from the desktop app\n * (no CORS issues) rather than a browser fetch.\n */\n\nimport { requestUrl, RequestUrlParam } from \"obsidian\";\nimport type { MicropubConfig, PublishResult } from \"./types\";\n\nexport class MicropubClient {\n  constructor(\n    private readonly getEndpoint: () => string,\n    private readonly getMediaEndpoint: () => string,\n    private readonly getToken: () => string,\n  ) {}\n\n  // \u2500\u2500 Config discovery \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  /** Fetch Micropub server config (syndication targets, media endpoint, etc.) */\n  async fetchConfig(): Promise<MicropubConfig> {\n    const url = `${this.getEndpoint()}?q=config`;\n    const resp = await requestUrl({\n      url,\n      method: \"GET\",\n      headers: this.authHeaders(),\n    });\n    return resp.json as MicropubConfig;\n  }\n\n  /**\n   * Discover micropub + token endpoint URLs from a site's home page\n   * by reading <link rel=\"micropub\"> and <link rel=\"token_endpoint\"> tags.\n   */\n  async discoverEndpoints(siteUrl: string): Promise<{\n    micropubEndpoint?: string;\n    tokenEndpoint?: string;\n    mediaEndpoint?: string;\n  }> {\n    const resp = await requestUrl({ url: siteUrl, method: \"GET\" });\n    const html = resp.text;\n\n    const micropub = this.extractLinkRel(html, \"micropub\");\n    const tokenEndpoint = this.extractLinkRel(html, \"token_endpoint\");\n\n    // After discovering the Micropub endpoint, fetch its config for the media URL\n    let mediaEndpoint: string | undefined;\n    if (micropub) {\n      try {\n        const cfg = await this.fetchConfigFrom(micropub);\n        mediaEndpoint = cfg[\"media-endpoint\"];\n      } catch {\n        // Non-fatal \u2014 media endpoint stays undefined\n      }\n    }\n\n    return { micropubEndpoint: micropub, tokenEndpoint, mediaEndpoint };\n  }\n\n  // \u2500\u2500 Post publishing \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  /**\n   * Create a new post via Micropub.\n   * Sends a JSON body with h-entry properties.\n   * Returns the Location header URL on success.\n   */\n  async createPost(properties: Record<string, unknown>): Promise<PublishResult> {\n    const body = {\n      type: [\"h-entry\"],\n      properties,\n    };\n\n    try {\n      const resp = await requestUrl({\n        url: this.getEndpoint(),\n        method: \"POST\",\n        headers: {\n          ...this.authHeaders(),\n          \"Content-Type\": \"application/json\",\n        },\n        body: JSON.stringify(body),\n        throw: false,\n      });\n\n      if (resp.status === 201 || resp.status === 202) {\n        const location =\n          resp.headers?.[\"location\"] ||\n          resp.headers?.[\"Location\"] ||\n          (resp.json as { url?: string })?.url;\n        return { success: true, url: location };\n      }\n\n      const detail = this.extractError(resp.text);\n      return { success: false, error: `HTTP ${resp.status}: ${detail}` };\n    } catch (err: unknown) {\n      return { success: false, error: String(err) };\n    }\n  }\n\n  /**\n   * Update an existing post.\n   * @param postUrl  The canonical URL of the post to update\n   * @param replace  Properties to replace (will overwrite existing values)\n   */\n  async updatePost(\n    postUrl: string,\n    replace: Record<string, unknown[]>,\n  ): Promise<PublishResult> {\n    const body = { action: \"update\", url: postUrl, replace };\n\n    try {\n      const resp = await requestUrl({\n        url: this.getEndpoint(),\n        method: \"POST\",\n        headers: {\n          ...this.authHeaders(),\n          \"Content-Type\": \"application/json\",\n        },\n        body: JSON.stringify(body),\n        throw: false,\n      });\n\n      if (resp.status >= 200 && resp.status < 300) {\n        return { success: true, url: postUrl };\n      }\n\n      return {\n        success: false,\n        error: `HTTP ${resp.status}: ${this.extractError(resp.text)}`,\n      };\n    } catch (err: unknown) {\n      return { success: false, error: String(err) };\n    }\n  }\n\n  // \u2500\u2500 Media upload \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  /**\n   * Upload a binary file to the media endpoint.\n   * @returns The URL of the uploaded media, or throws on failure.\n   */\n  async uploadMedia(\n    fileBuffer: ArrayBuffer,\n    fileName: string,\n    mimeType: string,\n  ): Promise<string> {\n    const endpoint = this.getMediaEndpoint() || `${this.getEndpoint()}/media`;\n\n    // Build multipart/form-data manually \u2014 Obsidian's requestUrl doesn't\n    // support FormData directly, so we encode the boundary ourselves.\n    const boundary = `----MicropubBoundary${Date.now()}`;\n    const header =\n      `--${boundary}\\r\\n` +\n      `Content-Disposition: form-data; name=\"file\"; filename=\"${fileName}\"\\r\\n` +\n      `Content-Type: ${mimeType}\\r\\n\\r\\n`;\n    const footer = `\\r\\n--${boundary}--\\r\\n`;\n\n    const headerBuf = new TextEncoder().encode(header);\n    const footerBuf = new TextEncoder().encode(footer);\n    const fileBuf = new Uint8Array(fileBuffer);\n\n    const combined = new Uint8Array(\n      headerBuf.length + fileBuf.length + footerBuf.length,\n    );\n    combined.set(headerBuf, 0);\n    combined.set(fileBuf, headerBuf.length);\n    combined.set(footerBuf, headerBuf.length + fileBuf.length);\n\n    const resp = await requestUrl({\n      url: endpoint,\n      method: \"POST\",\n      headers: {\n        ...this.authHeaders(),\n        \"Content-Type\": `multipart/form-data; boundary=${boundary}`,\n      },\n      body: combined.buffer,\n      throw: false,\n    });\n\n    if (resp.status === 201 || resp.status === 202) {\n      const location =\n        resp.headers?.[\"location\"] ||\n        resp.headers?.[\"Location\"] ||\n        (resp.json as { url?: string })?.url;\n      if (location) return location;\n    }\n\n    throw new Error(\n      `Media upload failed (HTTP ${resp.status}): ${this.extractError(resp.text)}`,\n    );\n  }\n\n  // \u2500\u2500 Helpers \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  private authHeaders(): Record<string, string> {\n    return { Authorization: `Bearer ${this.getToken()}` };\n  }\n\n  private extractLinkRel(html: string, rel: string): string | undefined {\n    // Match both <link> and HTTP Link headers embedded in HTML\n    const re = new RegExp(\n      `<link[^>]+rel=[\"']${rel}[\"'][^>]+href=[\"']([^\"']+)[\"']|<link[^>]+href=[\"']([^\"']+)[\"'][^>]+rel=[\"']${rel}[\"']`,\n      \"i\",\n    );\n    const m = html.match(re);\n    return m?.[1] ?? m?.[2];\n  }\n\n  private async fetchConfigFrom(endpoint: string): Promise<MicropubConfig> {\n    const resp = await requestUrl({\n      url: `${endpoint}?q=config`,\n      method: \"GET\",\n      headers: this.authHeaders(),\n    });\n    return resp.json as MicropubConfig;\n  }\n\n  private extractError(text: string): string {\n    try {\n      const obj = JSON.parse(text) as { error_description?: string; error?: string };\n      return obj.error_description ?? obj.error ?? text.slice(0, 200);\n    } catch {\n      return text.slice(0, 200);\n    }\n  }\n}\n", "/**\n * IndieAuth.ts \u2014 IndieAuth PKCE sign-in flow for obsidian-micropub\n *\n * Why no local HTTP server:\n *   IndieKit (and most IndieAuth servers) fetch the client_id URL server-side\n *   to retrieve app metadata. A local 127.0.0.1 address is unreachable from a\n *   remote server, so that approach always fails with \"fetch failed\".\n *\n * The solution \u2014 GitHub Pages relay:\n *   client_id  = https://svemagie.github.io/obsidian-micropub/\n *   redirect_uri = https://svemagie.github.io/obsidian-micropub/callback\n *\n *   Both are on the same host \u2192 IndieKit's host-matching check passes \u2713\n *   The callback page is a static HTML file that immediately redirects to\n *   obsidian://micropub-auth?code=CODE&state=STATE\n *   Obsidian's protocol handler (registered in main.ts) receives the code.\n *\n * Flow:\n *   1. Discover authorization_endpoint + token_endpoint from site HTML\n *   2. Generate PKCE code_verifier + code_challenge (SHA-256)\n *   3. Open browser \u2192 user's IndieAuth login page\n *   4. User logs in \u2192 server redirects to GitHub Pages callback\n *   5. Callback page redirects to obsidian://micropub-auth?code=...\n *   6. Plugin protocol handler resolves the pending Promise\n *   7. Exchange code for token at token_endpoint\n */\n\nimport * as crypto from \"crypto\";\nimport { requestUrl } from \"obsidian\";\n\nexport const CLIENT_ID   = \"https://svemagie.github.io/obsidian-micropub/\";\nexport const REDIRECT_URI = \"https://svemagie.github.io/obsidian-micropub/callback\";\n\nconst SCOPE = \"create update media\";\nconst AUTH_TIMEOUT_MS = 5 * 60 * 1000; // 5 minutes\n\nexport interface IndieAuthResult {\n  accessToken: string;\n  scope: string;\n  /** Canonical \"me\" URL returned by the token endpoint */\n  me: string;\n  authorizationEndpoint: string;\n  tokenEndpoint: string;\n  micropubEndpoint?: string;\n  mediaEndpoint?: string;\n}\n\nexport interface DiscoveredEndpoints {\n  authorizationEndpoint: string;\n  tokenEndpoint: string;\n  micropubEndpoint?: string;\n}\n\n/** Pending callback set by main.ts protocol handler */\nlet pendingCallback:\n  | { resolve: (params: Record<string, string>) => void; state: string }\n  | null = null;\n\n/**\n * Called by the Obsidian protocol handler in main.ts when\n * obsidian://micropub-auth is opened by the browser.\n */\nexport function handleProtocolCallback(params: Record<string, string>): void {\n  if (!pendingCallback) return;\n\n  const { resolve, state: expectedState } = pendingCallback;\n  pendingCallback = null;\n  resolve(params); // let signIn() validate state + extract code\n}\n\nexport class IndieAuth {\n  // \u2500\u2500 Public API \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  /**\n   * Discover IndieAuth + Micropub endpoint URLs from the site's home page\n   * by reading <link rel=\"...\"> tags in the HTML <head>.\n   */\n  static async discoverEndpoints(siteUrl: string): Promise<DiscoveredEndpoints> {\n    const resp = await requestUrl({ url: siteUrl, method: \"GET\" });\n    const html = resp.text;\n\n    const authorizationEndpoint = IndieAuth.extractLinkRel(html, \"authorization_endpoint\");\n    const tokenEndpoint         = IndieAuth.extractLinkRel(html, \"token_endpoint\");\n    const micropubEndpoint      = IndieAuth.extractLinkRel(html, \"micropub\");\n\n    if (!authorizationEndpoint) {\n      throw new Error(\n        `No <link rel=\"authorization_endpoint\"> found at ${siteUrl}. ` +\n        \"Make sure Indiekit is running and SITE_URL is set correctly.\",\n      );\n    }\n    if (!tokenEndpoint) {\n      throw new Error(`No <link rel=\"token_endpoint\"> found at ${siteUrl}.`);\n    }\n\n    return { authorizationEndpoint, tokenEndpoint, micropubEndpoint };\n  }\n\n  /**\n   * Run the full IndieAuth PKCE sign-in flow.\n   *\n   * Opens the browser at the user's IndieAuth login page. After login the\n   * browser is redirected to the GitHub Pages callback, which triggers\n   * the obsidian://micropub-auth protocol, which resolves the Promise here.\n   *\n   * Requires handleProtocolCallback() to be wired up in main.ts via\n   * this.registerObsidianProtocolHandler(\"micropub-auth\", handleProtocolCallback)\n   */\n  static async signIn(siteUrl: string): Promise<IndieAuthResult> {\n    // 1. Discover endpoints\n    const { authorizationEndpoint, tokenEndpoint, micropubEndpoint } =\n      await IndieAuth.discoverEndpoints(siteUrl);\n\n    // 2. Generate PKCE + state\n    const state        = IndieAuth.base64url(crypto.randomBytes(16));\n    const codeVerifier = IndieAuth.base64url(crypto.randomBytes(64));\n    const codeChallenge = IndieAuth.base64url(\n      crypto.createHash(\"sha256\").update(codeVerifier).digest(),\n    );\n\n    // 3. Register pending callback \u2014 resolved by handleProtocolCallback()\n    const callbackPromise = new Promise<Record<string, string>>(\n      (resolve, reject) => {\n        const timeout = setTimeout(() => {\n          pendingCallback = null;\n          reject(new Error(\"Sign-in timed out (5 min). Please try again.\"));\n        }, AUTH_TIMEOUT_MS);\n\n        pendingCallback = {\n          state,\n          resolve: (params) => {\n            clearTimeout(timeout);\n            resolve(params);\n          },\n        };\n      },\n    );\n\n    // 4. Build the authorization URL and open the browser\n    const authUrl = new URL(authorizationEndpoint);\n    authUrl.searchParams.set(\"response_type\",        \"code\");\n    authUrl.searchParams.set(\"client_id\",            CLIENT_ID);\n    authUrl.searchParams.set(\"redirect_uri\",         REDIRECT_URI);\n    authUrl.searchParams.set(\"state\",                state);\n    authUrl.searchParams.set(\"code_challenge\",       codeChallenge);\n    authUrl.searchParams.set(\"code_challenge_method\",\"S256\");\n    authUrl.searchParams.set(\"scope\",                SCOPE);\n    authUrl.searchParams.set(\"me\",                   siteUrl);\n\n    window.open(authUrl.toString());\n\n    // 5. Wait for obsidian://micropub-auth to be called\n    const callbackParams = await callbackPromise;\n\n    // 6. Validate state (CSRF protection)\n    if (callbackParams.state !== state) {\n      throw new Error(\"State mismatch \u2014 possible CSRF attack. Please try again.\");\n    }\n\n    const code = callbackParams.code;\n    if (!code) {\n      throw new Error(\n        callbackParams.error_description ??\n        callbackParams.error ??\n        \"No authorization code received.\",\n      );\n    }\n\n    // 7. Exchange code for token\n    const tokenResp = await requestUrl({\n      url: tokenEndpoint,\n      method: \"POST\",\n      headers: {\n        \"Content-Type\": \"application/x-www-form-urlencoded\",\n        Accept: \"application/json\",\n      },\n      body: new URLSearchParams({\n        grant_type:    \"authorization_code\",\n        code,\n        client_id:     CLIENT_ID,\n        redirect_uri:  REDIRECT_URI,\n        code_verifier: codeVerifier,\n      }).toString(),\n      throw: false,\n    });\n\n    const data = tokenResp.json as {\n      access_token?: string;\n      scope?: string;\n      me?: string;\n      error?: string;\n      error_description?: string;\n    };\n\n    if (!data.access_token) {\n      throw new Error(\n        data.error_description ??\n        data.error ??\n        `Token exchange failed (HTTP ${tokenResp.status})`,\n      );\n    }\n\n    return {\n      accessToken:           data.access_token,\n      scope:                 data.scope ?? SCOPE,\n      me:                    data.me ?? siteUrl,\n      authorizationEndpoint,\n      tokenEndpoint,\n      micropubEndpoint,\n    };\n  }\n\n  // \u2500\u2500 Helpers \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  private static base64url(buf: Buffer): string {\n    return buf.toString(\"base64\")\n      .replace(/\\+/g, \"-\")\n      .replace(/\\//g, \"_\")\n      .replace(/=/g, \"\");\n  }\n\n  static extractLinkRel(html: string, rel: string): string | undefined {\n    const re = new RegExp(\n      `<link[^>]+rel=[\"'][^\"']*\\\\b${rel}\\\\b[^\"']*[\"'][^>]+href=[\"']([^\"']+)[\"']` +\n      `|<link[^>]+href=[\"']([^\"']+)[\"'][^>]+rel=[\"'][^\"']*\\\\b${rel}\\\\b[^\"']*[\"']`,\n      \"i\",\n    );\n    const m = html.match(re);\n    return m?.[1] ?? m?.[2];\n  }\n}\n", "/**\n * Publisher.ts\n *\n * Orchestrates a full publish flow:\n *   1. Parse the active note's frontmatter + body\n *   2. Upload any local images to the media endpoint\n *   3. Build the Micropub properties object\n *   4. POST to the Micropub endpoint\n *   5. Optionally write the returned URL back to frontmatter\n *\n * Garden tag mapping:\n *   Obsidian tags #garden/plant \u2192 gardenStage: \"plant\" in properties\n *   The blog reads this as `gardenStage` frontmatter, so the Indiekit\n *   Micropub server must be configured to pass through unknown properties.\n */\n\nimport { App, TFile, parseFrontMatterAliases, parseYaml, stringifyYaml } from \"obsidian\";\nimport type { MicropubSettings, GardenStage, PublishResult } from \"./types\";\nimport { MicropubClient } from \"./MicropubClient\";\n\nconst GARDEN_TAG_PREFIX = \"garden/\";\n\nexport class Publisher {\n  private client: MicropubClient;\n\n  constructor(\n    private readonly app: App,\n    private readonly settings: MicropubSettings,\n  ) {\n    this.client = new MicropubClient(\n      () => settings.micropubEndpoint,\n      () => settings.mediaEndpoint,\n      () => settings.accessToken,\n    );\n  }\n\n  /** Publish the given file. Returns PublishResult. */\n  async publish(file: TFile): Promise<PublishResult> {\n    const raw = await this.app.vault.read(file);\n    const { frontmatter, body } = this.parseFrontmatter(raw);\n\n    // Determine if this is an update (post already has a URL) or new post\n    const existingUrl: string | undefined =\n      frontmatter[\"mp-url\"] ?? frontmatter[\"url\"] ?? undefined;\n\n    // Upload local images and rewrite markdown references\n    const { content: processedBody, uploadedUrls } =\n      await this.processImages(body);\n\n    // Build Micropub properties\n    const properties = this.buildProperties(frontmatter, processedBody, uploadedUrls);\n\n    let result: PublishResult;\n\n    if (existingUrl) {\n      // Update existing post\n      const replace: Record<string, unknown[]> = {};\n      for (const [k, v] of Object.entries(properties)) {\n        replace[k] = Array.isArray(v) ? v : [v];\n      }\n      result = await this.client.updatePost(existingUrl, replace);\n    } else {\n      // Create new post\n      result = await this.client.createPost(properties);\n    }\n\n    // Write URL back to frontmatter\n    if (result.success && result.url && this.settings.writeUrlToFrontmatter) {\n      await this.writeUrlToNote(file, raw, result.url);\n    }\n\n    return result;\n  }\n\n  // \u2500\u2500 Property builder \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  private buildProperties(\n    fm: Record<string, unknown>,\n    body: string,\n    uploadedUrls: string[],\n  ): Record<string, unknown> {\n    const props: Record<string, unknown> = {};\n\n    // \u2500\u2500 Post type detection \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    // Interaction posts (bookmark, like, reply, repost) have no body content.\n    // For those, only include content if the note body is non-empty (i.e. a comment/quote).\n    const trimmedBody = body.trim();\n\n    // \u2500\u2500 Interaction URL properties \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    // Support both camelCase (Obsidian-friendly) and hyphenated (Micropub-spec).\n    const bookmarkOf = fm[\"bookmarkOf\"] ?? fm[\"bookmark-of\"];\n    const likeOf     = fm[\"likeOf\"]     ?? fm[\"like-of\"];\n    const inReplyTo  = fm[\"inReplyTo\"]  ?? fm[\"in-reply-to\"];\n    const repostOf   = fm[\"repostOf\"]   ?? fm[\"repost-of\"];\n\n    if (bookmarkOf) props[\"bookmark-of\"] = [String(bookmarkOf)];\n    if (likeOf)     props[\"like-of\"]     = [String(likeOf)];\n    if (inReplyTo)  props[\"in-reply-to\"] = [String(inReplyTo)];\n    if (repostOf)   props[\"repost-of\"]   = [String(repostOf)];\n\n    // Content \u2014 omit for bare likes/reposts with no body text\n    const isInteractionWithoutBody =\n      (likeOf || repostOf) && !trimmedBody;\n    if (!isInteractionWithoutBody) {\n      props[\"content\"] = trimmedBody ? [{ html: trimmedBody }] : [{ html: \"\" }];\n    }\n\n    // \u2500\u2500 Standard properties \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n    // Title (articles have titles; notes/micro-posts don't)\n    if (fm[\"title\"]) {\n      props[\"name\"] = [String(fm[\"title\"])];\n    }\n\n    // Summary / excerpt\n    if (fm[\"summary\"] ?? fm[\"excerpt\"]) {\n      props[\"summary\"] = [String(fm[\"summary\"] ?? fm[\"excerpt\"])];\n    }\n\n    // Published date \u2014 prefer `created` (Obsidian default), fall back to `date`\n    const rawDate = fm[\"created\"] ?? fm[\"date\"];\n    if (rawDate) {\n      props[\"published\"] = [new Date(String(rawDate)).toISOString()];\n    }\n\n    // Categories from frontmatter `category` or `tags` (excluding garden/* tags)\n    const rawTags = this.resolveArray(fm[\"tags\"] ?? fm[\"category\"]);\n    const gardenStage = this.extractGardenStage(rawTags);\n    const normalTags = rawTags.filter(\n      (t) => !t.startsWith(GARDEN_TAG_PREFIX) && t !== \"garden\",\n    );\n    if (normalTags.length > 0) {\n      props[\"category\"] = normalTags;\n    }\n\n    // Garden stage \u2192 dedicated property\n    if (this.settings.mapGardenTags && gardenStage) {\n      props[\"garden-stage\"] = [gardenStage];\n    }\n\n    // Syndication targets\n    // Support both camelCase (mpSyndicateTo) used in existing blog posts and mp-syndicate-to\n    const syndicateTo = this.resolveArray(\n      fm[\"mp-syndicate-to\"] ?? fm[\"mpSyndicateTo\"],\n    );\n    const allSyndicateTo = [\n      ...new Set([...this.settings.defaultSyndicateTo, ...syndicateTo]),\n    ];\n    if (allSyndicateTo.length > 0) {\n      props[\"mp-syndicate-to\"] = allSyndicateTo;\n    }\n\n    // Visibility\n    const visibility =\n      (fm[\"visibility\"] as string) ?? this.settings.defaultVisibility;\n    if (visibility && visibility !== \"public\") {\n      props[\"visibility\"] = [visibility];\n    }\n\n    // AI disclosure (custom property passed through to Indiekit)\n    if (fm[\"ai\"] && typeof fm[\"ai\"] === \"object\") {\n      props[\"ai\"] = [fm[\"ai\"]];\n    }\n\n    // Photos: prefer structured photo array from frontmatter (with alt text),\n    // fall back to uploaded local images.\n    const fmPhotos = this.resolvePhotoArray(fm[\"photo\"]);\n    if (fmPhotos.length > 0) {\n      props[\"photo\"] = fmPhotos;\n    } else if (uploadedUrls.length > 0) {\n      props[\"photo\"] = uploadedUrls.map((url) => ({ value: url }));\n    }\n\n    // Pass through any `mp-*` properties from frontmatter verbatim\n    for (const [k, v] of Object.entries(fm)) {\n      if (k.startsWith(\"mp-\") && k !== \"mp-url\" && k !== \"mp-syndicate-to\") {\n        props[k] = this.resolveArray(v);\n      }\n    }\n\n    return props;\n  }\n\n  /**\n   * Normalise the `photo` frontmatter field into Micropub photo objects.\n   * Handles three formats:\n   *   - string URL: \"https://...\"\n   *   - array of strings: [\"https://...\"]\n   *   - array of objects: [{url: \"https://...\", alt: \"...\"}]\n   */\n  private resolvePhotoArray(\n    value: unknown,\n  ): Array<{ value: string; alt?: string }> {\n    if (!value) return [];\n    const items = Array.isArray(value) ? value : [value];\n    return items\n      .map((item) => {\n        if (typeof item === \"string\") return { value: item };\n        if (typeof item === \"object\" && item !== null) {\n          const obj = item as Record<string, unknown>;\n          const url = String(obj[\"url\"] ?? obj[\"value\"] ?? \"\");\n          if (!url) return null;\n          return obj[\"alt\"]\n            ? { value: url, alt: String(obj[\"alt\"]) }\n            : { value: url };\n        }\n        return null;\n      })\n      .filter((x): x is { value: string; alt?: string } => x !== null);\n  }\n\n  // \u2500\u2500 Garden tag extraction \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  /**\n   * Find the first #garden/<stage> tag and return the stage name.\n   * Supports both \"garden/plant\" (Obsidian array) and \"#garden/plant\" (inline).\n   */\n  private extractGardenStage(tags: string[]): GardenStage | undefined {\n    for (const tag of tags) {\n      const clean = tag.replace(/^#/, \"\");\n      if (clean.startsWith(GARDEN_TAG_PREFIX)) {\n        const stage = clean.slice(GARDEN_TAG_PREFIX.length) as GardenStage;\n        const valid: GardenStage[] = [\n          \"plant\", \"cultivate\", \"question\", \"repot\", \"revitalize\", \"revisit\",\n        ];\n        if (valid.includes(stage)) return stage;\n      }\n    }\n    return undefined;\n  }\n\n  // \u2500\u2500 Image processing \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  /**\n   * Find all `![[local-image.png]]` or `![alt](relative/path.jpg)` in the body,\n   * upload them to the media endpoint, and replace the references with remote URLs.\n   */\n  private async processImages(\n    body: string,\n  ): Promise<{ content: string; uploadedUrls: string[] }> {\n    const uploadedUrls: string[] = [];\n\n    // Match wiki-style embeds: ![[filename.ext]]\n    const wikiPattern = /!\\[\\[([^\\]]+\\.(png|jpg|jpeg|gif|webp|svg))\\]\\]/gi;\n    // Match markdown images: ![alt](path)\n    const mdPattern = /!\\[([^\\]]*)\\]\\(([^)]+\\.(png|jpg|jpeg|gif|webp|svg))\\)/gi;\n\n    let content = body;\n\n    // Process wiki-style embeds\n    const wikiMatches = [...body.matchAll(wikiPattern)];\n    for (const match of wikiMatches) {\n      const filename = match[1];\n      try {\n        const remoteUrl = await this.uploadLocalFile(filename);\n        if (remoteUrl) {\n          uploadedUrls.push(remoteUrl);\n          content = content.replace(match[0], `![${filename}](${remoteUrl})`);\n        }\n      } catch (err) {\n        console.warn(`[micropub] Failed to upload ${filename}:`, err);\n      }\n    }\n\n    // Process markdown image references\n    const mdMatches = [...content.matchAll(mdPattern)];\n    for (const match of mdMatches) {\n      const alt = match[1];\n      const path = match[2];\n      if (path.startsWith(\"http\")) continue; // already remote\n      try {\n        const remoteUrl = await this.uploadLocalFile(path);\n        if (remoteUrl) {\n          uploadedUrls.push(remoteUrl);\n          content = content.replace(match[0], `![${alt}](${remoteUrl})`);\n        }\n      } catch (err) {\n        console.warn(`[micropub] Failed to upload ${path}:`, err);\n      }\n    }\n\n    return { content, uploadedUrls };\n  }\n\n  private async uploadLocalFile(path: string): Promise<string | undefined> {\n    const file = this.app.vault.getFiles().find(\n      (f) => f.name === path || f.path === path,\n    );\n    if (!file) return undefined;\n\n    const buffer = await this.app.vault.readBinary(file);\n    const mimeType = this.guessMimeType(file.extension);\n\n    return this.client.uploadMedia(buffer, file.name, mimeType);\n  }\n\n  // \u2500\u2500 Frontmatter helpers \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  private parseFrontmatter(raw: string): {\n    frontmatter: Record<string, unknown>;\n    body: string;\n  } {\n    const fmMatch = raw.match(/^---\\r?\\n([\\s\\S]*?)\\r?\\n---\\r?\\n([\\s\\S]*)$/);\n    if (!fmMatch) return { frontmatter: {}, body: raw };\n\n    let frontmatter: Record<string, unknown> = {};\n    try {\n      frontmatter = (parseYaml(fmMatch[1]) ?? {}) as Record<string, unknown>;\n    } catch {\n      // Malformed frontmatter \u2014 treat as empty\n    }\n\n    return { frontmatter, body: fmMatch[2] };\n  }\n\n  private async writeUrlToNote(\n    file: TFile,\n    originalContent: string,\n    url: string,\n  ): Promise<void> {\n    const fmMatch = originalContent.match(\n      /^(---\\r?\\n[\\s\\S]*?\\r?\\n---\\r?\\n)([\\s\\S]*)$/,\n    );\n\n    if (!fmMatch) {\n      // No existing frontmatter \u2014 prepend it\n      const newFm = `---\\nmp-url: \"${url}\"\\n---\\n`;\n      await this.app.vault.modify(file, newFm + originalContent);\n      return;\n    }\n\n    // Inject mp-url into existing frontmatter block\n    const fmBlock = fmMatch[1];\n    const body = fmMatch[2];\n\n    if (fmBlock.includes(\"mp-url:\")) {\n      // Replace existing mp-url line\n      const updated = fmBlock.replace(\n        /mp-url:.*(\\r?\\n)/,\n        `mp-url: \"${url}\"$1`,\n      );\n      await this.app.vault.modify(file, updated + body);\n    } else {\n      // Insert mp-url before closing ---\n      const updated = fmBlock.replace(\n        /(\\r?\\n---\\r?\\n)$/,\n        `\\nmp-url: \"${url}\"$1`,\n      );\n      await this.app.vault.modify(file, updated + body);\n    }\n  }\n\n  private resolveArray(value: unknown): string[] {\n    if (!value) return [];\n    if (Array.isArray(value)) return value.map(String);\n    return [String(value)];\n  }\n\n  private guessMimeType(ext: string): string {\n    const map: Record<string, string> = {\n      png: \"image/png\",\n      jpg: \"image/jpeg\",\n      jpeg: \"image/jpeg\",\n      gif: \"image/gif\",\n      webp: \"image/webp\",\n      svg: \"image/svg+xml\",\n    };\n    return map[ext.toLowerCase()] ?? \"application/octet-stream\";\n  }\n}\n"],
  "mappings": "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",
  "names": ["import_obsidian", "import_obsidian", "import_obsidian", "import_obsidian"]
}

diff --git a/package-lock.json b/package-lock.json
deleted file mode 100644
index c6f17b8..0000000
--- a/package-lock.json
+++ /dev/null
@@ -1,676 +0,0 @@
-{
- "name": "obsidian-micropub",
- "version": "0.1.0",
- "lockfileVersion": 3,
- "requires": true,
- "packages": {
- "": {
- "name": "obsidian-micropub",
- "version": "0.1.0",
- "license": "MIT",
- "devDependencies": {
- "@types/node": "^20.0.0",
- "@types/qrcode": "^1.5.5",
- "builtin-modules": "^4.0.0",
- "esbuild": "^0.25.0",
- "obsidian": "latest",
- "tslib": "^2.6.0",
- "typescript": "^5.0.0"
- }
- },
- "node_modules/@codemirror/state": {
- "version": "6.5.0",
- "resolved": "https://registry.npmjs.org/@codemirror/state/-/state-6.5.0.tgz",
- "integrity": "sha512-MwBHVK60IiIHDcoMet78lxt6iw5gJOGSbNbOIVBHWVXIH4/Nq1+GQgLLGgI1KlnN86WDXsPudVaqYHKBIx7Eyw==",
- "dev": true,
- "license": "MIT",
- "peer": true,
- "dependencies": {
- "@marijn/find-cluster-break": "^1.0.0"
- }
- },
- "node_modules/@codemirror/view": {
- "version": "6.38.6",
- "resolved": "https://registry.npmjs.org/@codemirror/view/-/view-6.38.6.tgz",
- "integrity": "sha512-qiS0z1bKs5WOvHIAC0Cybmv4AJSkAXgX5aD6Mqd2epSLlVJsQl8NG23jCVouIgkh4All/mrbdsf2UOLFnJw0tw==",
- "dev": true,
- "license": "MIT",
- "peer": true,
- "dependencies": {
- "@codemirror/state": "^6.5.0",
- "crelt": "^1.0.6",
- "style-mod": "^4.1.0",
- "w3c-keyname": "^2.2.4"
- }
- },
- "node_modules/@esbuild/aix-ppc64": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.25.12.tgz",
- "integrity": "sha512-Hhmwd6CInZ3dwpuGTF8fJG6yoWmsToE+vYgD4nytZVxcu1ulHpUQRAB1UJ8+N1Am3Mz4+xOByoQoSZf4D+CpkA==",
- "cpu": [
- "ppc64"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "aix"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/android-arm": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.25.12.tgz",
- "integrity": "sha512-VJ+sKvNA/GE7Ccacc9Cha7bpS8nyzVv0jdVgwNDaR4gDMC/2TTRc33Ip8qrNYUcpkOHUT5OZ0bUcNNVZQ9RLlg==",
- "cpu": [
- "arm"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "android"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/android-arm64": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.25.12.tgz",
- "integrity": "sha512-6AAmLG7zwD1Z159jCKPvAxZd4y/VTO0VkprYy+3N2FtJ8+BQWFXU+OxARIwA46c5tdD9SsKGZ/1ocqBS/gAKHg==",
- "cpu": [
- "arm64"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "android"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/android-x64": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.25.12.tgz",
- "integrity": "sha512-5jbb+2hhDHx5phYR2By8GTWEzn6I9UqR11Kwf22iKbNpYrsmRB18aX/9ivc5cabcUiAT/wM+YIZ6SG9QO6a8kg==",
- "cpu": [
- "x64"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "android"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/darwin-arm64": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.25.12.tgz",
- "integrity": "sha512-N3zl+lxHCifgIlcMUP5016ESkeQjLj/959RxxNYIthIg+CQHInujFuXeWbWMgnTo4cp5XVHqFPmpyu9J65C1Yg==",
- "cpu": [
- "arm64"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "darwin"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/darwin-x64": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.25.12.tgz",
- "integrity": "sha512-HQ9ka4Kx21qHXwtlTUVbKJOAnmG1ipXhdWTmNXiPzPfWKpXqASVcWdnf2bnL73wgjNrFXAa3yYvBSd9pzfEIpA==",
- "cpu": [
- "x64"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "darwin"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/freebsd-arm64": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.25.12.tgz",
- "integrity": "sha512-gA0Bx759+7Jve03K1S0vkOu5Lg/85dou3EseOGUes8flVOGxbhDDh/iZaoek11Y8mtyKPGF3vP8XhnkDEAmzeg==",
- "cpu": [
- "arm64"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "freebsd"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/freebsd-x64": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.25.12.tgz",
- "integrity": "sha512-TGbO26Yw2xsHzxtbVFGEXBFH0FRAP7gtcPE7P5yP7wGy7cXK2oO7RyOhL5NLiqTlBh47XhmIUXuGciXEqYFfBQ==",
- "cpu": [
- "x64"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "freebsd"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/linux-arm": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.25.12.tgz",
- "integrity": "sha512-lPDGyC1JPDou8kGcywY0YILzWlhhnRjdof3UlcoqYmS9El818LLfJJc3PXXgZHrHCAKs/Z2SeZtDJr5MrkxtOw==",
- "cpu": [
- "arm"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "linux"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/linux-arm64": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.25.12.tgz",
- "integrity": "sha512-8bwX7a8FghIgrupcxb4aUmYDLp8pX06rGh5HqDT7bB+8Rdells6mHvrFHHW2JAOPZUbnjUpKTLg6ECyzvas2AQ==",
- "cpu": [
- "arm64"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "linux"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/linux-ia32": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.25.12.tgz",
- "integrity": "sha512-0y9KrdVnbMM2/vG8KfU0byhUN+EFCny9+8g202gYqSSVMonbsCfLjUO+rCci7pM0WBEtz+oK/PIwHkzxkyharA==",
- "cpu": [
- "ia32"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "linux"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/linux-loong64": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.25.12.tgz",
- "integrity": "sha512-h///Lr5a9rib/v1GGqXVGzjL4TMvVTv+s1DPoxQdz7l/AYv6LDSxdIwzxkrPW438oUXiDtwM10o9PmwS/6Z0Ng==",
- "cpu": [
- "loong64"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "linux"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/linux-mips64el": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.25.12.tgz",
- "integrity": "sha512-iyRrM1Pzy9GFMDLsXn1iHUm18nhKnNMWscjmp4+hpafcZjrr2WbT//d20xaGljXDBYHqRcl8HnxbX6uaA/eGVw==",
- "cpu": [
- "mips64el"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "linux"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/linux-ppc64": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.25.12.tgz",
- "integrity": "sha512-9meM/lRXxMi5PSUqEXRCtVjEZBGwB7P/D4yT8UG/mwIdze2aV4Vo6U5gD3+RsoHXKkHCfSxZKzmDssVlRj1QQA==",
- "cpu": [
- "ppc64"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "linux"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/linux-riscv64": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.25.12.tgz",
- "integrity": "sha512-Zr7KR4hgKUpWAwb1f3o5ygT04MzqVrGEGXGLnj15YQDJErYu/BGg+wmFlIDOdJp0PmB0lLvxFIOXZgFRrdjR0w==",
- "cpu": [
- "riscv64"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "linux"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/linux-s390x": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.25.12.tgz",
- "integrity": "sha512-MsKncOcgTNvdtiISc/jZs/Zf8d0cl/t3gYWX8J9ubBnVOwlk65UIEEvgBORTiljloIWnBzLs4qhzPkJcitIzIg==",
- "cpu": [
- "s390x"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "linux"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/linux-x64": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.25.12.tgz",
- "integrity": "sha512-uqZMTLr/zR/ed4jIGnwSLkaHmPjOjJvnm6TVVitAa08SLS9Z0VM8wIRx7gWbJB5/J54YuIMInDquWyYvQLZkgw==",
- "cpu": [
- "x64"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "linux"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/netbsd-arm64": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/netbsd-arm64/-/netbsd-arm64-0.25.12.tgz",
- "integrity": "sha512-xXwcTq4GhRM7J9A8Gv5boanHhRa/Q9KLVmcyXHCTaM4wKfIpWkdXiMog/KsnxzJ0A1+nD+zoecuzqPmCRyBGjg==",
- "cpu": [
- "arm64"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "netbsd"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/netbsd-x64": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.25.12.tgz",
- "integrity": "sha512-Ld5pTlzPy3YwGec4OuHh1aCVCRvOXdH8DgRjfDy/oumVovmuSzWfnSJg+VtakB9Cm0gxNO9BzWkj6mtO1FMXkQ==",
- "cpu": [
- "x64"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "netbsd"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/openbsd-arm64": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/openbsd-arm64/-/openbsd-arm64-0.25.12.tgz",
- "integrity": "sha512-fF96T6KsBo/pkQI950FARU9apGNTSlZGsv1jZBAlcLL1MLjLNIWPBkj5NlSz8aAzYKg+eNqknrUJ24QBybeR5A==",
- "cpu": [
- "arm64"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "openbsd"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/openbsd-x64": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.25.12.tgz",
- "integrity": "sha512-MZyXUkZHjQxUvzK7rN8DJ3SRmrVrke8ZyRusHlP+kuwqTcfWLyqMOE3sScPPyeIXN/mDJIfGXvcMqCgYKekoQw==",
- "cpu": [
- "x64"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "openbsd"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/openharmony-arm64": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/openharmony-arm64/-/openharmony-arm64-0.25.12.tgz",
- "integrity": "sha512-rm0YWsqUSRrjncSXGA7Zv78Nbnw4XL6/dzr20cyrQf7ZmRcsovpcRBdhD43Nuk3y7XIoW2OxMVvwuRvk9XdASg==",
- "cpu": [
- "arm64"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "openharmony"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/sunos-x64": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.25.12.tgz",
- "integrity": "sha512-3wGSCDyuTHQUzt0nV7bocDy72r2lI33QL3gkDNGkod22EsYl04sMf0qLb8luNKTOmgF/eDEDP5BFNwoBKH441w==",
- "cpu": [
- "x64"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "sunos"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/win32-arm64": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.25.12.tgz",
- "integrity": "sha512-rMmLrur64A7+DKlnSuwqUdRKyd3UE7oPJZmnljqEptesKM8wx9J8gx5u0+9Pq0fQQW8vqeKebwNXdfOyP+8Bsg==",
- "cpu": [
- "arm64"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "win32"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/win32-ia32": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.25.12.tgz",
- "integrity": "sha512-HkqnmmBoCbCwxUKKNPBixiWDGCpQGVsrQfJoVGYLPT41XWF8lHuE5N6WhVia2n4o5QK5M4tYr21827fNhi4byQ==",
- "cpu": [
- "ia32"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "win32"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@esbuild/win32-x64": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.25.12.tgz",
- "integrity": "sha512-alJC0uCZpTFrSL0CCDjcgleBXPnCrEAhTBILpeAp7M/OFgoqtAetfBzX0xM00MUsVVPpVjlPuMbREqnZCXaTnA==",
- "cpu": [
- "x64"
- ],
- "dev": true,
- "license": "MIT",
- "optional": true,
- "os": [
- "win32"
- ],
- "engines": {
- "node": ">=18"
- }
- },
- "node_modules/@marijn/find-cluster-break": {
- "version": "1.0.2",
- "resolved": "https://registry.npmjs.org/@marijn/find-cluster-break/-/find-cluster-break-1.0.2.tgz",
- "integrity": "sha512-l0h88YhZFyKdXIFNfSWpyjStDjGHwZ/U7iobcK1cQQD8sejsONdQtTVU+1wVN1PBw40PiiHB1vA5S7VTfQiP9g==",
- "dev": true,
- "license": "MIT",
- "peer": true
- },
- "node_modules/@types/codemirror": {
- "version": "5.60.8",
- "resolved": "https://registry.npmjs.org/@types/codemirror/-/codemirror-5.60.8.tgz",
- "integrity": "sha512-VjFgDF/eB+Aklcy15TtOTLQeMjTo07k7KAjql8OK5Dirr7a6sJY4T1uVBDuTVG9VEmn1uUsohOpYnVfgC6/jyw==",
- "dev": true,
- "license": "MIT",
- "dependencies": {
- "@types/tern": "*"
- }
- },
- "node_modules/@types/estree": {
- "version": "1.0.8",
- "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz",
- "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==",
- "dev": true,
- "license": "MIT"
- },
- "node_modules/@types/node": {
- "version": "20.19.37",
- "resolved": "https://registry.npmjs.org/@types/node/-/node-20.19.37.tgz",
- "integrity": "sha512-8kzdPJ3FsNsVIurqBs7oodNnCEVbni9yUEkaHbgptDACOPW04jimGagZ51E6+lXUwJjgnBw+hyko/lkFWCldqw==",
- "dev": true,
- "license": "MIT",
- "dependencies": {
- "undici-types": "~6.21.0"
- }
- },
- "node_modules/@types/qrcode": {
- "version": "1.5.6",
- "resolved": "https://registry.npmjs.org/@types/qrcode/-/qrcode-1.5.6.tgz",
- "integrity": "sha512-te7NQcV2BOvdj2b1hCAHzAoMNuj65kNBMz0KBaxM6c3VGBOhU0dURQKOtH8CFNI/dsKkwlv32p26qYQTWoB5bw==",
- "dev": true,
- "license": "MIT",
- "dependencies": {
- "@types/node": "*"
- }
- },
- "node_modules/@types/tern": {
- "version": "0.23.9",
- "resolved": "https://registry.npmjs.org/@types/tern/-/tern-0.23.9.tgz",
- "integrity": "sha512-ypzHFE/wBzh+BlH6rrBgS5I/Z7RD21pGhZ2rltb/+ZrVM1awdZwjx7hE5XfuYgHWk9uvV5HLZN3SloevCAp3Bw==",
- "dev": true,
- "license": "MIT",
- "dependencies": {
- "@types/estree": "*"
- }
- },
- "node_modules/builtin-modules": {
- "version": "4.0.0",
- "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-4.0.0.tgz",
- "integrity": "sha512-p1n8zyCkt1BVrKNFymOHjcDSAl7oq/gUvfgULv2EblgpPVQlQr9yHnWjg9IJ2MhfwPqiYqMMrr01OY7yQoK2yA==",
- "dev": true,
- "license": "MIT",
- "engines": {
- "node": ">=18.20"
- },
- "funding": {
- "url": "https://github.com/sponsors/sindresorhus"
- }
- },
- "node_modules/crelt": {
- "version": "1.0.6",
- "resolved": "https://registry.npmjs.org/crelt/-/crelt-1.0.6.tgz",
- "integrity": "sha512-VQ2MBenTq1fWZUH9DJNGti7kKv6EeAuYr3cLwxUWhIu1baTaXh4Ib5W2CqHVqib4/MqbYGJqiL3Zb8GJZr3l4g==",
- "dev": true,
- "license": "MIT",
- "peer": true
- },
- "node_modules/esbuild": {
- "version": "0.25.12",
- "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.25.12.tgz",
- "integrity": "sha512-bbPBYYrtZbkt6Os6FiTLCTFxvq4tt3JKall1vRwshA3fdVztsLAatFaZobhkBC8/BrPetoa0oksYoKXoG4ryJg==",
- "dev": true,
- "hasInstallScript": true,
- "license": "MIT",
- "bin": {
- "esbuild": "bin/esbuild"
- },
- "engines": {
- "node": ">=18"
- },
- "optionalDependencies": {
- "@esbuild/aix-ppc64": "0.25.12",
- "@esbuild/android-arm": "0.25.12",
- "@esbuild/android-arm64": "0.25.12",
- "@esbuild/android-x64": "0.25.12",
- "@esbuild/darwin-arm64": "0.25.12",
- "@esbuild/darwin-x64": "0.25.12",
- "@esbuild/freebsd-arm64": "0.25.12",
- "@esbuild/freebsd-x64": "0.25.12",
- "@esbuild/linux-arm": "0.25.12",
- "@esbuild/linux-arm64": "0.25.12",
- "@esbuild/linux-ia32": "0.25.12",
- "@esbuild/linux-loong64": "0.25.12",
- "@esbuild/linux-mips64el": "0.25.12",
- "@esbuild/linux-ppc64": "0.25.12",
- "@esbuild/linux-riscv64": "0.25.12",
- "@esbuild/linux-s390x": "0.25.12",
- "@esbuild/linux-x64": "0.25.12",
- "@esbuild/netbsd-arm64": "0.25.12",
- "@esbuild/netbsd-x64": "0.25.12",
- "@esbuild/openbsd-arm64": "0.25.12",
- "@esbuild/openbsd-x64": "0.25.12",
- "@esbuild/openharmony-arm64": "0.25.12",
- "@esbuild/sunos-x64": "0.25.12",
- "@esbuild/win32-arm64": "0.25.12",
- "@esbuild/win32-ia32": "0.25.12",
- "@esbuild/win32-x64": "0.25.12"
- }
- },
- "node_modules/moment": {
- "version": "2.29.4",
- "resolved": "https://registry.npmjs.org/moment/-/moment-2.29.4.tgz",
- "integrity": "sha512-5LC9SOxjSc2HF6vO2CyuTDNivEdoz2IvyJJGj6X8DJ0eFyfszE0QiEd+iXmBvUP3WHxSjFH/vIsA0EN00cgr8w==",
- "dev": true,
- "license": "MIT",
- "engines": {
- "node": "*"
- }
- },
- "node_modules/obsidian": {
- "version": "1.12.3",
- "resolved": "https://registry.npmjs.org/obsidian/-/obsidian-1.12.3.tgz",
- "integrity": "sha512-HxWqe763dOqzXjnNiHmAJTRERN8KILBSqxDSEqbeSr7W8R8Jxezzbca+nz1LiiqXnMpM8lV2jzAezw3CZ4xNUw==",
- "dev": true,
- "license": "MIT",
- "dependencies": {
- "@types/codemirror": "5.60.8",
- "moment": "2.29.4"
- },
- "peerDependencies": {
- "@codemirror/state": "6.5.0",
- "@codemirror/view": "6.38.6"
- }
- },
- "node_modules/style-mod": {
- "version": "4.1.3",
- "resolved": "https://registry.npmjs.org/style-mod/-/style-mod-4.1.3.tgz",
- "integrity": "sha512-i/n8VsZydrugj3Iuzll8+x/00GH2vnYsk1eomD8QiRrSAeW6ItbCQDtfXCeJHd0iwiNagqjQkvpvREEPtW3IoQ==",
- "dev": true,
- "license": "MIT",
- "peer": true
- },
- "node_modules/tslib": {
- "version": "2.8.1",
- "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz",
- "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==",
- "dev": true,
- "license": "0BSD"
- },
- "node_modules/typescript": {
- "version": "5.9.3",
- "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.3.tgz",
- "integrity": "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==",
- "dev": true,
- "license": "Apache-2.0",
- "bin": {
- "tsc": "bin/tsc",
- "tsserver": "bin/tsserver"
- },
- "engines": {
- "node": ">=14.17"
- }
- },
- "node_modules/undici-types": {
- "version": "6.21.0",
- "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.21.0.tgz",
- "integrity": "sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ==",
- "dev": true,
- "license": "MIT"
- },
- "node_modules/w3c-keyname": {
- "version": "2.2.8",
- "resolved": "https://registry.npmjs.org/w3c-keyname/-/w3c-keyname-2.2.8.tgz",
- "integrity": "sha512-dpojBhNsCNN7T82Tm7k26A6G9ML3NkhDsnw9n/eoxSRlVBB4CEtIQ/KTCLI2Fwf3ataSXRhYFkQi3SlnFwPvPQ==",
- "dev": true,
- "license": "MIT",
- "peer": true
- }
- }
-}