From 8fbf85d86baa23f065e9bf63bd8aef607609392c Mon Sep 17 00:00:00 2001 From: Hongbo Wu Date: Mon, 20 Feb 2023 23:14:52 +0800 Subject: [PATCH] fix: use omnivore title as the filename and add omnivore id to the frontmatter which is used to deduplicate files with the same name --- package-lock.json | 4 +-- src/main.ts | 91 ++++++++++++++++++++++++++++++++++++++++------- src/util.ts | 7 +++- 3 files changed, 86 insertions(+), 16 deletions(-) diff --git a/package-lock.json b/package-lock.json index d518777..7a44046 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,12 +1,12 @@ { "name": "obsidian-omnivore", - "version": "1.0.0", + "version": "1.0.4", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "obsidian-omnivore", - "version": "1.0.0", + "version": "1.0.4", "license": "MIT", "dependencies": { "diff-match-patch": "^1.0.5", diff --git a/src/main.ts b/src/main.ts index 2e154ae..b2fa1e2 100644 --- a/src/main.ts +++ b/src/main.ts @@ -8,6 +8,7 @@ import { Plugin, PluginSettingTab, Setting, + stringifyYaml, TFile, TFolder, } from "obsidian"; @@ -19,7 +20,7 @@ import { loadArticles, PageType, parseDateTime, - unicodeSlug, + replaceIllegalChars, } from "./util"; import { FolderSuggest } from "./settings/file-suggest"; @@ -57,6 +58,7 @@ const DEFAULT_SETTINGS: Settings = { syncAt: "", customQuery: "", template: `--- +id: {{{id}}} title: {{{title}}} {{#author}} author: {{{author}}} @@ -179,7 +181,7 @@ export default class OmnivorePlugin extends Plugin { await this.saveSettings(); try { - console.log(`obsidian-omnivore starting sync since: '${syncAt}`); + console.log(`obsidian-omnivore starting sync since: '${syncAt}'`); new Notice("🚀 Fetching articles ..."); @@ -215,11 +217,6 @@ export default class OmnivorePlugin extends Plugin { await this.app.vault.createFolder(folderName); } - // use unicode slug to show characters from other languages in the file name - const pageName = `${folderName}/${unicodeSlug( - article.title, - article.savedAt - )}.md`; const siteName = article.siteName || this.siteNameFromUrl(article.originalArticleUrl); @@ -255,6 +252,7 @@ export default class OmnivorePlugin extends Plugin { // Build content string based on template const content = Mustache.render(template, { + id: article.id, title: article.title, omnivoreUrl: `https://omnivore.app/me/${article.slug}`, siteName, @@ -270,15 +268,82 @@ export default class OmnivorePlugin extends Plugin { content: article.content, }); + // add frontmatter to the content + const frontmatter = { + id: article.id, + title: article.title, + author: article.author, + tags: article.labels?.map((l) => l.name), + date_saved: dateSaved, + }; + // remove null and empty values from frontmatter + const filteredFrontmatter = Object.fromEntries( + Object.entries(frontmatter).filter( + ([_, value]) => value != null && value !== "" + ) + ); + const frontmatterYaml = stringifyYaml(filteredFrontmatter); + const frontmatterString = `---\n${frontmatterYaml}---`; + // Modify the contents of the note with the updated front matter + const updatedContent = content.replace( + /^(---[\s\S]*?---)/gm, + frontmatterString + ); + // use the title as the filename + const pageName = `${folderName}/${replaceIllegalChars( + article.title + )}.md`; const normalizedPath = normalizePath(pageName); const omnivoreFile = app.vault.getAbstractFileByPath(normalizedPath); - if (omnivoreFile instanceof TFile) { - const existingContent = await this.app.vault.read(omnivoreFile); - if (existingContent !== content) { - await this.app.vault.modify(omnivoreFile, content); + try { + if (omnivoreFile instanceof TFile) { + await app.fileManager.processFrontMatter( + omnivoreFile, + async (frontMatter) => { + const id = frontMatter.id; + if (id && id !== article.id) { + // this article has the same name but different id + const newPageName = `${folderName}/${replaceIllegalChars( + article.title + )}-${article.id}.md`; + const newNormalizedPath = normalizePath(newPageName); + const newOmnivoreFile = + app.vault.getAbstractFileByPath(newNormalizedPath); + if (newOmnivoreFile instanceof TFile) { + // a file with the same name and id already exists, so we need to update it + const existingContent = await this.app.vault.read( + newOmnivoreFile + ); + if (existingContent !== updatedContent) { + await this.app.vault.modify( + newOmnivoreFile, + updatedContent + ); + } + return; + } + // a file with the same name but different id already exists, so we need to create it + await this.app.vault.create( + newNormalizedPath, + updatedContent + ); + return; + } + // a file with the same id already exists, so we might need to update it + const existingContent = await this.app.vault.read( + omnivoreFile + ); + if (existingContent !== updatedContent) { + await this.app.vault.modify(omnivoreFile, updatedContent); + } + } + ); + } else if (!omnivoreFile) { + // file doesn't exist, so we need to create it + await this.app.vault.create(normalizedPath, updatedContent); } - } else if (!omnivoreFile) { - await this.app.vault.create(normalizedPath, content); + } catch (e) { + console.error(e); } } } diff --git a/src/util.ts b/src/util.ts index cec47aa..55cb307 100644 --- a/src/util.ts +++ b/src/util.ts @@ -53,6 +53,7 @@ export enum PageType { } export interface Article { + id: string; title: string; siteName: string; originalArticleUrl: string; @@ -119,7 +120,7 @@ export const loadArticles = async ( const res = await requestUrl({ url: endpoint, headers: requestHeaders(apiKey), - body: `{"query":"\\n query Search($after: String, $first: Int, $query: String, $includeContent: Boolean, $format: String) {\\n search(first: $first, after: $after, query: $query, includeContent: $includeContent, format: $format) {\\n ... on SearchSuccess {\\n edges {\\n node {\\n title\\n slug\\n siteName\\n originalArticleUrl\\n url\\n author\\n updatedAt\\n description\\n savedAt\\n pageType\\n content\\n highlights {\\n id\\n quote\\n annotation\\n patch\\n updatedAt\\n }\\n labels {\\n name\\n }\\n }\\n }\\n pageInfo {\\n hasNextPage\\n }\\n }\\n ... on SearchError {\\n errorCodes\\n }\\n }\\n }\\n ","variables":{"after":"${after}","first":${first}, "query":"${ + body: `{"query":"\\n query Search($after: String, $first: Int, $query: String, $includeContent: Boolean, $format: String) {\\n search(first: $first, after: $after, query: $query, includeContent: $includeContent, format: $format) {\\n ... on SearchSuccess {\\n edges {\\n node {\\n title\\n slug\\n siteName\\n originalArticleUrl\\n url\\n author\\n updatedAt\\n description\\n savedAt\\n pageType\\n content\\n id\\n highlights {\\n id\\n quote\\n annotation\\n patch\\n updatedAt\\n }\\n labels {\\n name\\n }\\n }\\n }\\n pageInfo {\\n hasNextPage\\n }\\n }\\n ... on SearchError {\\n errorCodes\\n }\\n }\\n }\\n ","variables":{"after":"${after}","first":${first}, "query":"${ updatedAt ? "updated:" + updatedAt : "" } sort:saved-asc ${query}", "includeContent": ${includeContent}, "format": "${format}"}}`, method: "POST", @@ -227,3 +228,7 @@ export const unicodeSlug = (str: string, savedAt: string) => { new Date(savedAt).getTime().toString(16) ); }; + +export const replaceIllegalChars = (str: string): string => { + return str.replace(/[/\\?%*:|"<>]/g, "-"); +};