From 21a398f1adb8a545f2ace99aefa0469c2c2ed04e Mon Sep 17 00:00:00 2001 From: svemagie <869694+svemagie@users.noreply.github.com> Date: Sat, 14 Mar 2026 18:10:58 +0100 Subject: [PATCH] fix: use `created` property as published date (Obsidian default), fall back to `date` --- main.js | 15 ++++++++------- src/Publisher.ts | 7 ++++--- 2 files changed, 12 insertions(+), 10 deletions(-) diff --git a/main.js b/main.js index 8b930a2..d66563d 100644 --- a/main.js +++ b/main.js @@ -595,7 +595,7 @@ var Publisher = class { } // ── Property builder ───────────────────────────────────────────────────── buildProperties(fm, body, uploadedUrls) { - var _a, _b, _c, _d, _e, _f, _g, _h, _i; + 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"]; @@ -616,10 +616,11 @@ var Publisher = class { if ((_e = fm["summary"]) != null ? _e : fm["excerpt"]) { props["summary"] = [String((_f = fm["summary"]) != null ? _f : fm["excerpt"])]; } - if (fm["date"]) { - props["published"] = [new Date(String(fm["date"])).toISOString()]; + const rawDate = (_g = fm["created"]) != null ? _g : fm["date"]; + if (rawDate) { + props["published"] = [new Date(String(rawDate)).toISOString()]; } - const rawTags = this.resolveArray((_g = fm["tags"]) != null ? _g : fm["category"]); + 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" @@ -631,7 +632,7 @@ var Publisher = class { props["garden-stage"] = [gardenStage]; } const syndicateTo = this.resolveArray( - (_h = fm["mp-syndicate-to"]) != null ? _h : fm["mpSyndicateTo"] + (_i = fm["mp-syndicate-to"]) != null ? _i : fm["mpSyndicateTo"] ); const allSyndicateTo = [ .../* @__PURE__ */ new Set([...this.settings.defaultSyndicateTo, ...syndicateTo]) @@ -639,7 +640,7 @@ var Publisher = class { if (allSyndicateTo.length > 0) { props["mp-syndicate-to"] = allSyndicateTo; } - const visibility = (_i = fm["visibility"]) != null ? _i : this.settings.defaultVisibility; + const visibility = (_j = fm["visibility"]) != null ? _j : this.settings.defaultVisibility; if (visibility && visibility !== "public") { props["visibility"] = [visibility]; } @@ -904,4 +905,4 @@ ${result.url}` : ""; 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\n    if (fm[\"date\"]) {\n      props[\"published\"] = [new Date(String(fm[\"date\"])).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"]
}
 +//# 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/src/Publisher.ts b/src/Publisher.ts index c3e51bf..bea73b6 100644 --- a/src/Publisher.ts +++ b/src/Publisher.ts @@ -117,9 +117,10 @@ export class Publisher { props["summary"] = [String(fm["summary"] ?? fm["excerpt"])]; } - // Published date - if (fm["date"]) { - props["published"] = [new Date(String(fm["date"])).toISOString()]; + // Published date — prefer `created` (Obsidian default), fall back to `date` + const rawDate = fm["created"] ?? fm["date"]; + if (rawDate) { + props["published"] = [new Date(String(rawDate)).toISOString()]; } // Categories from frontmatter `category` or `tags` (excluding garden/* tags)