From 69ea5efb7c16fbb63f1c66074dcd4c60251935c4 Mon Sep 17 00:00:00 2001 From: Mara Date: Mon, 3 Apr 2023 11:25:09 +0200 Subject: [PATCH 01/11] feat: add cut-off frontmatter and cut-off text on %%---%% see #142 --- main.js | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/main.js b/main.js index e789c1ca..132885a4 100644 --- a/main.js +++ b/main.js @@ -9,6 +9,7 @@ const DEFAULT_SETTINGS = { header_exclusions: "", path_only: "", show_full_path: false, + cut_off_frontmatter: false, expanded_view: true, group_nearest_by_file: false, language: "en", @@ -2476,6 +2477,7 @@ class SmartConnectionsSettingsTab extends Obsidian.PluginSettingTab { const self_ref_pronouns_list = containerEl.createEl("span", { text: this.get_self_ref_list() }); + new Obsidian.Setting(containerEl).setName("Cut off frontmatter").setDesc("Cut off frontmatter in the prompt to gain characters in reply generation").addToggle((toggle) => { toggle.setValue(this.plugin.settings.cut_off_frontmatter).onChange(async (value) => { this.plugin.settings.cut_off_frontmatter = value; await this.plugin.saveSettings(); }); }); containerEl.createEl("h2", { text: "Exclusions" }); @@ -3494,6 +3496,7 @@ class SmartConnectionsChatModel { // max chars for this note is max_chars divided by number of notes left const this_max_chars = (notes.length - i > 1) ? Math.floor(max_chars / (notes.length - i)) : max_chars; const note_content = await this.get_note_contents(notes[i], {char_limit: this_max_chars}); + console.log(note_content); system_input += `---BEGIN NOTE: [[${notes[i].basename}]]---\n` system_input += note_content; system_input += `---END NOTE---\n` @@ -3539,11 +3542,17 @@ class SmartConnectionsChatModel { if(!(note instanceof Obsidian.TFile)) return ""; // get file content let file_content = await this.app.vault.cachedRead(note); + //cut off front matter + if (this.plugin.settings.cut_off_frontmatter) { + file_content = file_content.replace(/\s*---[\s\S]*?---/,""); + } // check if contains dataview code block if(file_content.indexOf("```dataview") > -1){ // if contains dataview code block get all dataview code blocks file_content = await this.render_dataview_queries(file_content, note.path, opts); } + // cut off note if %%---%% is found + file_content = file_content.replace(/%%---%%[\s\S]*/, ""); return file_content.substring(0, opts.char_limit); } From 492662fb5ff5082c5d4ec6787134831d9fead009 Mon Sep 17 00:00:00 2001 From: Mara Date: Mon, 3 Apr 2023 11:26:50 +0200 Subject: [PATCH 02/11] feat: add cut-off frontmatter and cut-off text on %%---%% see #142 --- main.js | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/main.js b/main.js index e789c1ca..132885a4 100644 --- a/main.js +++ b/main.js @@ -9,6 +9,7 @@ const DEFAULT_SETTINGS = { header_exclusions: "", path_only: "", show_full_path: false, + cut_off_frontmatter: false, expanded_view: true, group_nearest_by_file: false, language: "en", @@ -2476,6 +2477,7 @@ class SmartConnectionsSettingsTab extends Obsidian.PluginSettingTab { const self_ref_pronouns_list = containerEl.createEl("span", { text: this.get_self_ref_list() }); + new Obsidian.Setting(containerEl).setName("Cut off frontmatter").setDesc("Cut off frontmatter in the prompt to gain characters in reply generation").addToggle((toggle) => { toggle.setValue(this.plugin.settings.cut_off_frontmatter).onChange(async (value) => { this.plugin.settings.cut_off_frontmatter = value; await this.plugin.saveSettings(); }); }); containerEl.createEl("h2", { text: "Exclusions" }); @@ -3494,6 +3496,7 @@ class SmartConnectionsChatModel { // max chars for this note is max_chars divided by number of notes left const this_max_chars = (notes.length - i > 1) ? Math.floor(max_chars / (notes.length - i)) : max_chars; const note_content = await this.get_note_contents(notes[i], {char_limit: this_max_chars}); + console.log(note_content); system_input += `---BEGIN NOTE: [[${notes[i].basename}]]---\n` system_input += note_content; system_input += `---END NOTE---\n` @@ -3539,11 +3542,17 @@ class SmartConnectionsChatModel { if(!(note instanceof Obsidian.TFile)) return ""; // get file content let file_content = await this.app.vault.cachedRead(note); + //cut off front matter + if (this.plugin.settings.cut_off_frontmatter) { + file_content = file_content.replace(/\s*---[\s\S]*?---/,""); + } // check if contains dataview code block if(file_content.indexOf("```dataview") > -1){ // if contains dataview code block get all dataview code blocks file_content = await this.render_dataview_queries(file_content, note.path, opts); } + // cut off note if %%---%% is found + file_content = file_content.replace(/%%---%%[\s\S]*/, ""); return file_content.substring(0, opts.char_limit); } From 6f9ad55c31a0a524fee6cc84e2372bee1ab4c856 Mon Sep 17 00:00:00 2001 From: Mara Date: Sat, 22 Apr 2023 19:06:46 +0200 Subject: [PATCH 03/11] feat: open in main view instead of right panel --- main.js | 33 ++++++++++++++++++++++++++++----- 1 file changed, 28 insertions(+), 5 deletions(-) diff --git a/main.js b/main.js index 852077ca..2b75db81 100644 --- a/main.js +++ b/main.js @@ -22,6 +22,7 @@ const DEFAULT_SETTINGS = { smart_chat_model: "gpt-3.5-turbo", view_open: true, version: "", + open_in_big_view: false, }; const MAX_EMBED_STRING_LENGTH = 25000; @@ -35,26 +36,31 @@ const SMART_TRANSLATION = { "pronous": ["my", "I", "me", "mine", "our", "ours", "us", "we"], "prompt": "Based on your notes", "initial_message": "Hi, I'm ChatGPT with access to your notes via Smart Connections. Ask me a question about your notes and I'll try to answer it.", + "try_placeholder": `Try "Based on my notes" or "Summarize [[this note]]" or "Important tasks in /folder/"` }, "es": { "pronous": ["mi", "yo", "mí", "tú"], "prompt": "Basándose en sus notas", "initial_message": "Hola, soy ChatGPT con acceso a tus apuntes a través de Smart Connections. Hazme una pregunta sobre tus apuntes e intentaré responderte.", + "try_placeholder": `Prueba "Basado en mis notas" o "Resumen [[esta nota]]" o "Tareas importantes en /carpeta/"` }, "fr": { "pronous": ["me", "mon", "ma", "mes", "moi", "nous", "notre", "nos", "je", "j'", "m'"], "prompt": "D'après vos notes", "initial_message": "Bonjour, je suis ChatGPT et j'ai accès à vos notes via Smart Connections. Posez-moi une question sur vos notes et j'essaierai d'y répondre.", + "try_placeholder": `Essayez "D'après mes notes" ou "Résume [[cette note]]" ou "Tâches importantes dans /dossier/"` }, "de": { "pronous": ["mein", "meine", "meinen", "meiner", "meines", "mir", "uns", "unser", "unseren", "unserer", "unseres"], "prompt": "Basierend auf Ihren Notizen", "initial_message": "Hallo, ich bin ChatGPT und habe über Smart Connections Zugang zu Ihren Notizen. Stellen Sie mir eine Frage zu Ihren Notizen und ich werde versuchen, sie zu beantworten.", + "try_placeholder": `Versuchen Sie "Basierend auf meinen Notizen" oder "Zusammenfassen [[dieser Notiz]]" oder "Wichtige Aufgaben im /Ordner/"` }, "it": { "pronous": ["mio", "mia", "miei", "mie", "noi", "nostro", "nostri", "nostra", "nostre"], "prompt": "Sulla base degli appunti", "initial_message": "Ciao, sono ChatGPT e ho accesso ai tuoi appunti tramite Smart Connections. Fatemi una domanda sui vostri appunti e cercherò di rispondervi.", + "try_placeholder": `Prova "Sulla base dei miei appunti" o "Riassumi [[questo appunto]]" o "Compiti importanti in /cartella/"` }, } @@ -336,10 +342,17 @@ class SmartConnectionsPlugin extends Obsidian.Plugin { // open chat view async open_chat() { this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE); - await this.app.workspace.getRightLeaf(false).setViewState({ - type: SMART_CONNECTIONS_CHAT_VIEW_TYPE, - active: true, - }); + if (!this.settings.open_in_big_view) { + await this.app.workspace.getRightLeaf(false).setViewState({ + type: SMART_CONNECTIONS_CHAT_VIEW_TYPE, + active: true, + }); + } else { + await this.app.workspace.getLeaf(true).setViewState({ + type: SMART_CONNECTIONS_CHAT_VIEW_TYPE, + active: true, + }) + } this.app.workspace.revealLeaf( this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE)[0] ); @@ -2609,6 +2622,16 @@ class SmartConnectionsSettingsTab extends Obsidian.PluginSettingTab { this.plugin.settings.chat_open = value; await this.plugin.saveSettings(true); })); + // open in big view or small view + new Obsidian.Setting(containerEl).setName("open_in_big_view").setDesc("Open in big view or small view.").addDropdown((dropdown) => { + dropdown.addOption(false, "Right pane (small)"); + dropdown.addOption(true, "Main pane (big)"); + dropdown.setValue(this.plugin.settings.open_in_big_view); + dropdown.onChange(async (value) => { + this.plugin.settings.open_in_big_view = value; + await this.plugin.saveSettings(true); + }); + }); containerEl.createEl("h2", { text: "Advanced" }); @@ -2901,7 +2924,7 @@ class SmartConnectionsChatView extends Obsidian.ItemView { this.textarea = chat_input.createEl("textarea", { cls: "sc-chat-input", attr: { - placeholder: `Try "Based on my notes" or "Summarize [[this note]]" or "Important tasks in /folder/"` + placeholder: SMART_TRANSLATION[this.plugin.settings.language].try_placeholder } }); // use contenteditable instead of textarea From da2b6992b88c241f817c8833a75ca99f04fe77fc Mon Sep 17 00:00:00 2001 From: Mara Date: Sat, 22 Apr 2023 19:24:05 +0200 Subject: [PATCH 04/11] fix: sconvert string to bool --- main.js | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/main.js b/main.js index 2b75db81..a07fb55d 100644 --- a/main.js +++ b/main.js @@ -342,7 +342,7 @@ class SmartConnectionsPlugin extends Obsidian.Plugin { // open chat view async open_chat() { this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE); - if (!this.settings.open_in_big_view) { + if (this.settings.open_in_big_view === "false") { await this.app.workspace.getRightLeaf(false).setViewState({ type: SMART_CONNECTIONS_CHAT_VIEW_TYPE, active: true, @@ -2630,6 +2630,8 @@ class SmartConnectionsSettingsTab extends Obsidian.PluginSettingTab { dropdown.onChange(async (value) => { this.plugin.settings.open_in_big_view = value; await this.plugin.saveSettings(true); + this.plugin.open_chat(); + }); }); containerEl.createEl("h2", { From a57f5cce7fd94c65d68b2bc1b561997e4bc7c438 Mon Sep 17 00:00:00 2001 From: Mara Date: Sat, 22 Apr 2023 19:24:05 +0200 Subject: [PATCH 05/11] fix: sconvert string to bool --- main.js | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/main.js b/main.js index 2b75db81..a07fb55d 100644 --- a/main.js +++ b/main.js @@ -342,7 +342,7 @@ class SmartConnectionsPlugin extends Obsidian.Plugin { // open chat view async open_chat() { this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE); - if (!this.settings.open_in_big_view) { + if (this.settings.open_in_big_view === "false") { await this.app.workspace.getRightLeaf(false).setViewState({ type: SMART_CONNECTIONS_CHAT_VIEW_TYPE, active: true, @@ -2630,6 +2630,8 @@ class SmartConnectionsSettingsTab extends Obsidian.PluginSettingTab { dropdown.onChange(async (value) => { this.plugin.settings.open_in_big_view = value; await this.plugin.saveSettings(true); + this.plugin.open_chat(); + }); }); containerEl.createEl("h2", { From 90fb71950767160bb8483863a75d6c3b48d21ba2 Mon Sep 17 00:00:00 2001 From: Mara Date: Sat, 22 Apr 2023 19:58:35 +0200 Subject: [PATCH 06/11] fix: convert string to bool --- main.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/main.js b/main.js index a07fb55d..9caff50b 100644 --- a/main.js +++ b/main.js @@ -342,7 +342,7 @@ class SmartConnectionsPlugin extends Obsidian.Plugin { // open chat view async open_chat() { this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE); - if (this.settings.open_in_big_view === "false") { + if (!this.settings.open_in_big_view) { await this.app.workspace.getRightLeaf(false).setViewState({ type: SMART_CONNECTIONS_CHAT_VIEW_TYPE, active: true, @@ -2628,7 +2628,7 @@ class SmartConnectionsSettingsTab extends Obsidian.PluginSettingTab { dropdown.addOption(true, "Main pane (big)"); dropdown.setValue(this.plugin.settings.open_in_big_view); dropdown.onChange(async (value) => { - this.plugin.settings.open_in_big_view = value; + this.plugin.settings.open_in_big_view = JSON.parse(value); await this.plugin.saveSettings(true); this.plugin.open_chat(); From 99def4f106c9600bca32d02c7acac39341578cb6 Mon Sep 17 00:00:00 2001 From: Mara Date: Tue, 2 Jan 2024 13:02:27 +0100 Subject: [PATCH 07/11] test --- data.json | 25 + dist/main.js | 3329 ---------------------------------------------- main.js | 3382 +++++++++++++++++++++++++++++++++++++++++++++++ package.json | 3 - pnpm-lock.yaml | 240 ++++ src/index.js | 415 +++++- src/vec_lite.js | 347 +++++ styles.css | 206 +++ 8 files changed, 4592 insertions(+), 3355 deletions(-) create mode 100644 data.json delete mode 100644 dist/main.js create mode 100644 main.js create mode 100644 pnpm-lock.yaml create mode 100644 src/vec_lite.js create mode 100644 styles.css diff --git a/data.json b/data.json new file mode 100644 index 00000000..f592d2cd --- /dev/null +++ b/data.json @@ -0,0 +1,25 @@ +{ + "api_key": "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", + "chat_open": true, + "file_exclusions": "sortspec", + "folder_exclusions": "", + "header_exclusions": "", + "path_only": "", + "show_full_path": false, + "cut_off_frontmatter": true, + "expanded_view": true, + "group_nearest_by_file": false, + "language": "fr", + "log_render": false, + "log_render_files": true, + "recently_sent_retry_notice": false, + "skip_sections": false, + "smart_chat_model": "gpt-4-1106-preview", + "view_open": true, + "version": "1.6.47", + "open_in_big_view": false, + "results_count": 30, + "failed_files": [ + "" + ] +} \ No newline at end of file diff --git a/dist/main.js b/dist/main.js deleted file mode 100644 index c301650a..00000000 --- a/dist/main.js +++ /dev/null @@ -1,3329 +0,0 @@ -var __getOwnPropNames = Object.getOwnPropertyNames; -var __commonJS = (cb, mod) => function __require() { - return mod || (0, cb[__getOwnPropNames(cb)[0]])((mod = { exports: {} }).exports, mod), mod.exports; -}; - -// ../vec_lite/vec_lite.js -var require_vec_lite = __commonJS({ - "../vec_lite/vec_lite.js"(exports2, module2) { - var VecLite2 = class { - constructor(config) { - this.config = { - file_name: "embeddings-3.json", - folder_path: ".vec_lite", - exists_adapter: null, - mkdir_adapter: null, - read_adapter: null, - rename_adapter: null, - stat_adapter: null, - write_adapter: null, - ...config - }; - this.file_name = this.config.file_name; - this.folder_path = config.folder_path; - this.file_path = this.folder_path + "/" + this.file_name; - this.embeddings = false; - } - async file_exists(path) { - if (this.config.exists_adapter) { - return await this.config.exists_adapter(path); - } else { - throw new Error("exists_adapter not set"); - } - } - async mkdir(path) { - if (this.config.mkdir_adapter) { - return await this.config.mkdir_adapter(path); - } else { - throw new Error("mkdir_adapter not set"); - } - } - async read_file(path) { - if (this.config.read_adapter) { - return await this.config.read_adapter(path); - } else { - throw new Error("read_adapter not set"); - } - } - async rename(old_path, new_path) { - if (this.config.rename_adapter) { - return await this.config.rename_adapter(old_path, new_path); - } else { - throw new Error("rename_adapter not set"); - } - } - async stat(path) { - if (this.config.stat_adapter) { - return await this.config.stat_adapter(path); - } else { - throw new Error("stat_adapter not set"); - } - } - async write_file(path, data) { - if (this.config.write_adapter) { - return await this.config.write_adapter(path, data); - } else { - throw new Error("write_adapter not set"); - } - } - async load(retries = 0) { - try { - const embeddings_file = await this.read_file(this.file_path); - this.embeddings = JSON.parse(embeddings_file); - console.log("loaded embeddings file: " + this.file_path); - return true; - } catch (error) { - if (retries < 3) { - console.log("retrying load()"); - await new Promise((r) => setTimeout(r, 1e3 + 1e3 * retries)); - return await this.load(retries + 1); - } - console.log("failed to load embeddings file, prompt user to initiate bulk embed"); - return false; - } - } - async init_embeddings_file() { - if (!await this.file_exists(this.folder_path)) { - await this.mkdir(this.folder_path); - console.log("created folder: " + this.folder_path); - } else { - console.log("folder already exists: " + this.folder_path); - } - if (!await this.file_exists(this.file_path)) { - await this.write_file(this.file_path, "{}"); - console.log("created embeddings file: " + this.file_path); - } else { - console.log("embeddings file already exists: " + this.file_path); - } - } - async save() { - const embeddings = JSON.stringify(this.embeddings); - const embeddings_file_exists = await this.file_exists(this.file_path); - if (embeddings_file_exists) { - const new_file_size = embeddings.length; - const existing_file_size = await this.stat(this.file_path).then((stat) => stat.size); - if (new_file_size > existing_file_size * 0.5) { - await this.write_file(this.file_path, embeddings); - console.log("embeddings file size: " + new_file_size + " bytes"); - } else { - const warning_message = [ - "Warning: New embeddings file size is significantly smaller than existing embeddings file size.", - "Aborting to prevent possible loss of embeddings data.", - "New file size: " + new_file_size + " bytes.", - "Existing file size: " + existing_file_size + " bytes.", - "Restarting Obsidian may fix this." - ]; - console.log(warning_message.join(" ")); - await this.write_file(this.folder_path + "/unsaved-embeddings.json", embeddings); - throw new Error("Error: New embeddings file size is significantly smaller than existing embeddings file size. Aborting to prevent possible loss of embeddings data."); - } - } else { - await this.init_embeddings_file(); - return await this.save(); - } - return true; - } - cos_sim(vector1, vector2) { - let dotProduct = 0; - let normA = 0; - let normB = 0; - for (let i = 0; i < vector1.length; i++) { - dotProduct += vector1[i] * vector2[i]; - normA += vector1[i] * vector1[i]; - normB += vector2[i] * vector2[i]; - } - if (normA === 0 || normB === 0) { - return 0; - } else { - return dotProduct / (Math.sqrt(normA) * Math.sqrt(normB)); - } - } - nearest(to_vec, filter = {}) { - filter = { - results_count: 30, - ...filter - }; - let nearest = []; - const from_keys = Object.keys(this.embeddings); - for (let i = 0; i < from_keys.length; i++) { - if (filter.skip_sections) { - const from_path = this.embeddings[from_keys[i]].meta.path; - if (from_path.indexOf("#") > -1) - continue; - } - if (filter.skip_key) { - if (filter.skip_key === from_keys[i]) - continue; - if (filter.skip_key === this.embeddings[from_keys[i]].meta.parent) - continue; - } - if (filter.path_begins_with) { - if (typeof filter.path_begins_with === "string" && !this.embeddings[from_keys[i]].meta.path.startsWith(filter.path_begins_with)) - continue; - if (Array.isArray(filter.path_begins_with) && !filter.path_begins_with.some((path) => this.embeddings[from_keys[i]].meta.path.startsWith(path))) - continue; - } - nearest.push({ - link: this.embeddings[from_keys[i]].meta.path, - similarity: this.cos_sim(to_vec, this.embeddings[from_keys[i]].vec), - size: this.embeddings[from_keys[i]].meta.size - }); - } - nearest.sort(function(a, b) { - return b.similarity - a.similarity; - }); - nearest = nearest.slice(0, filter.results_count); - return nearest; - } - find_nearest_embeddings(to_vec, filter = {}) { - const default_filter = { - max: this.max_sources - }; - filter = { ...default_filter, ...filter }; - if (Array.isArray(to_vec) && to_vec.length !== this.vec_len) { - this.nearest = {}; - for (let i = 0; i < to_vec.length; i++) { - this.find_nearest_embeddings(to_vec[i], { - max: Math.floor(filter.max / to_vec.length) - }); - } - } else { - const from_keys = Object.keys(this.embeddings); - for (let i = 0; i < from_keys.length; i++) { - if (this.validate_type(this.embeddings[from_keys[i]])) - continue; - const sim = this.computeCosineSimilarity(to_vec, this.embeddings[from_keys[i]].vec); - if (this.nearest[from_keys[i]]) { - this.nearest[from_keys[i]] += sim; - } else { - this.nearest[from_keys[i]] = sim; - } - } - } - let nearest = Object.keys(this.nearest).map((key) => { - return { - key, - similarity: this.nearest[key] - }; - }); - nearest = this.sort_by_similarity(nearest); - nearest = nearest.slice(0, filter.max); - nearest = nearest.map((item) => { - return { - link: this.embeddings[item.key].meta.path, - similarity: item.similarity, - len: this.embeddings[item.key].meta.len || this.embeddings[item.key].meta.size - }; - }); - return nearest; - } - sort_by_similarity(nearest) { - return nearest.sort(function(a, b) { - const a_score = a.similarity; - const b_score = b.similarity; - if (a_score > b_score) - return -1; - if (a_score < b_score) - return 1; - return 0; - }); - } - // check if key from embeddings exists in files - clean_up_embeddings(files) { - console.log("cleaning up embeddings"); - const keys = Object.keys(this.embeddings); - let deleted_embeddings = 0; - for (const key of keys) { - const path = this.embeddings[key].meta.path; - if (!files.find((file) => path.startsWith(file.path))) { - delete this.embeddings[key]; - deleted_embeddings++; - continue; - } - if (path.indexOf("#") > -1) { - const parent_key = this.embeddings[key].meta.parent; - if (!this.embeddings[parent_key]) { - delete this.embeddings[key]; - deleted_embeddings++; - continue; - } - if (!this.embeddings[parent_key].meta) { - delete this.embeddings[key]; - deleted_embeddings++; - continue; - } - if (this.embeddings[parent_key].meta.children && this.embeddings[parent_key].meta.children.indexOf(key) < 0) { - delete this.embeddings[key]; - deleted_embeddings++; - continue; - } - } - } - return { deleted_embeddings, total_embeddings: keys.length }; - } - get(key) { - return this.embeddings[key] || null; - } - get_meta(key) { - const embedding = this.get(key); - if (embedding && embedding.meta) { - return embedding.meta; - } - return null; - } - get_mtime(key) { - const meta = this.get_meta(key); - if (meta && meta.mtime) { - return meta.mtime; - } - return null; - } - get_hash(key) { - const meta = this.get_meta(key); - if (meta && meta.hash) { - return meta.hash; - } - return null; - } - get_size(key) { - const meta = this.get_meta(key); - if (meta && meta.size) { - return meta.size; - } - return null; - } - get_children(key) { - const meta = this.get_meta(key); - if (meta && meta.children) { - return meta.children; - } - return null; - } - get_vec(key) { - const embedding = this.get(key); - if (embedding && embedding.vec) { - return embedding.vec; - } - return null; - } - save_embedding(key, vec, meta) { - this.embeddings[key] = { - vec, - meta - }; - } - mtime_is_current(key, source_mtime) { - const mtime = this.get_mtime(key); - if (mtime && mtime >= source_mtime) { - return true; - } - return false; - } - async force_refresh() { - this.embeddings = null; - this.embeddings = {}; - let current_datetime = Math.floor(Date.now() / 1e3); - await this.rename(this.file_path, this.folder_path + "/embeddings-" + current_datetime + ".json"); - await this.init_embeddings_file(); - } - }; - module2.exports = VecLite2; - } -}); - -// src/index.js -var Obsidian = require("obsidian"); -var VecLite = require_vec_lite(); -var DEFAULT_SETTINGS = { - api_key: "", - chat_open: true, - file_exclusions: "", - folder_exclusions: "", - header_exclusions: "", - path_only: "", - show_full_path: false, - expanded_view: true, - group_nearest_by_file: false, - language: "en", - log_render: false, - log_render_files: false, - recently_sent_retry_notice: false, - skip_sections: false, - smart_chat_model: "gpt-3.5-turbo-16k", - view_open: true, - version: "" -}; -var MAX_EMBED_STRING_LENGTH = 25e3; -var VERSION; -var SUPPORTED_FILE_TYPES = ["md", "canvas"]; -var SMART_TRANSLATION = { - "en": { - "pronous": ["my", "I", "me", "mine", "our", "ours", "us", "we"], - "prompt": "Based on your notes", - "initial_message": "Hi, I'm ChatGPT with access to your notes via Smart Connections. Ask me a question about your notes and I'll try to answer it." - }, - "es": { - "pronous": ["mi", "yo", "m\xED", "t\xFA"], - "prompt": "Bas\xE1ndose en sus notas", - "initial_message": "Hola, soy ChatGPT con acceso a tus apuntes a trav\xE9s de Smart Connections. Hazme una pregunta sobre tus apuntes e intentar\xE9 responderte." - }, - "fr": { - "pronous": ["me", "mon", "ma", "mes", "moi", "nous", "notre", "nos", "je", "j'", "m'"], - "prompt": "D'apr\xE8s vos notes", - "initial_message": "Bonjour, je suis ChatGPT et j'ai acc\xE8s \xE0 vos notes via Smart Connections. Posez-moi une question sur vos notes et j'essaierai d'y r\xE9pondre." - }, - "de": { - "pronous": ["mein", "meine", "meinen", "meiner", "meines", "mir", "uns", "unser", "unseren", "unserer", "unseres"], - "prompt": "Basierend auf Ihren Notizen", - "initial_message": "Hallo, ich bin ChatGPT und habe \xFCber Smart Connections Zugang zu Ihren Notizen. Stellen Sie mir eine Frage zu Ihren Notizen und ich werde versuchen, sie zu beantworten." - }, - "it": { - "pronous": ["mio", "mia", "miei", "mie", "noi", "nostro", "nostri", "nostra", "nostre"], - "prompt": "Sulla base degli appunti", - "initial_message": "Ciao, sono ChatGPT e ho accesso ai tuoi appunti tramite Smart Connections. Fatemi una domanda sui vostri appunti e cercher\xF2 di rispondervi." - } -}; -var crypto = require("crypto"); -function md5(str) { - return crypto.createHash("md5").update(str).digest("hex"); -} -var SmartConnectionsPlugin = class extends Obsidian.Plugin { - // constructor - constructor() { - super(...arguments); - this.api = null; - this.embeddings_loaded = false; - this.file_exclusions = []; - this.folders = []; - this.has_new_embeddings = false; - this.header_exclusions = []; - this.nearest_cache = {}; - this.path_only = []; - this.render_log = {}; - this.render_log.deleted_embeddings = 0; - this.render_log.exclusions_logs = {}; - this.render_log.failed_embeddings = []; - this.render_log.files = []; - this.render_log.new_embeddings = 0; - this.render_log.skipped_low_delta = {}; - this.render_log.token_usage = 0; - this.render_log.tokens_saved_by_cache = 0; - this.retry_notice_timeout = null; - this.save_timeout = null; - this.sc_branding = {}; - this.self_ref_kw_regex = null; - this.update_available = false; - } - async onload() { - this.app.workspace.onLayoutReady(this.initialize.bind(this)); - } - onunload() { - this.output_render_log(); - console.log("unloading plugin"); - this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE); - this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE); - } - async initialize() { - console.log("Loading Smart Connections plugin"); - VERSION = this.manifest.version; - await this.loadSettings(); - setTimeout(this.check_for_update.bind(this), 3e3); - setInterval(this.check_for_update.bind(this), 108e5); - this.addIcon(); - this.addCommand({ - id: "sc-find-notes", - name: "Find: Make Smart Connections", - icon: "pencil_icon", - hotkeys: [], - // editorCallback: async (editor) => { - editorCallback: async (editor) => { - if (editor.somethingSelected()) { - let selected_text = editor.getSelection(); - await this.make_connections(selected_text); - } else { - this.nearest_cache = {}; - await this.make_connections(); - } - } - }); - this.addCommand({ - id: "smart-connections-view", - name: "Open: View Smart Connections", - callback: () => { - this.open_view(); - } - }); - this.addCommand({ - id: "smart-connections-chat", - name: "Open: Smart Chat Conversation", - callback: () => { - this.open_chat(); - } - }); - this.addCommand({ - id: "smart-connections-random", - name: "Open: Random Note from Smart Connections", - callback: () => { - this.open_random_note(); - } - }); - this.addSettingTab(new SmartConnectionsSettingsTab(this.app, this)); - this.registerView(SMART_CONNECTIONS_VIEW_TYPE, (leaf) => new SmartConnectionsView(leaf, this)); - this.registerView(SMART_CONNECTIONS_CHAT_VIEW_TYPE, (leaf) => new SmartConnectionsChatView(leaf, this)); - this.registerMarkdownCodeBlockProcessor("smart-connections", this.render_code_block.bind(this)); - if (this.settings.view_open) { - this.open_view(); - } - if (this.settings.chat_open) { - this.open_chat(); - } - if (this.settings.version !== VERSION) { - this.settings.version = VERSION; - await this.saveSettings(); - this.open_view(); - } - this.add_to_gitignore(); - this.api = new ScSearchApi(this.app, this); - (window["SmartSearchApi"] = this.api) && this.register(() => delete window["SmartSearchApi"]); - } - async init_vecs() { - this.smart_vec_lite = new VecLite({ - folder_path: ".smart-connections", - exists_adapter: this.app.vault.adapter.exists.bind(this.app.vault.adapter), - mkdir_adapter: this.app.vault.adapter.mkdir.bind(this.app.vault.adapter), - read_adapter: this.app.vault.adapter.read.bind(this.app.vault.adapter), - rename_adapter: this.app.vault.adapter.rename.bind(this.app.vault.adapter), - stat_adapter: this.app.vault.adapter.stat.bind(this.app.vault.adapter), - write_adapter: this.app.vault.adapter.write.bind(this.app.vault.adapter) - }); - this.embeddings_loaded = await this.smart_vec_lite.load(); - return this.embeddings_loaded; - } - async update_to_v2() { - if (!this.settings.license_key) - return new Obsidian.Notice("[Smart Connections] Supporter license key required for early access to V2"); - const v2 = await (0, Obsidian.requestUrl)({ - url: "https://sync.smartconnections.app/download_v2", - method: "POST", - headers: { - "Content-Type": "application/json" - }, - body: JSON.stringify({ - license_key: this.settings.license_key - }) - }); - if (v2.status !== 200) - return console.error("Error downloading version 2", v2); - console.log(v2); - await this.app.vault.adapter.write(".obsidian/plugins/smart-connections/main.js", v2.json.main); - await this.app.vault.adapter.write(".obsidian/plugins/smart-connections/manifest.json", v2.json.manifest); - await this.app.vault.adapter.write(".obsidian/plugins/smart-connections/styles.css", v2.json.styles); - window.restart_plugin = async (id) => { - console.log("restarting plugin", id); - await window.app.plugins.disablePlugin(id); - await window.app.plugins.enablePlugin(id); - console.log("plugin restarted", id); - }; - window.restart_plugin(this.manifest.id); - } - async loadSettings() { - this.settings = Object.assign({}, DEFAULT_SETTINGS, await this.loadData()); - if (this.settings.file_exclusions && this.settings.file_exclusions.length > 0) { - this.file_exclusions = this.settings.file_exclusions.split(",").map((file) => { - return file.trim(); - }); - } - if (this.settings.folder_exclusions && this.settings.folder_exclusions.length > 0) { - const folder_exclusions = this.settings.folder_exclusions.split(",").map((folder) => { - folder = folder.trim(); - if (folder.slice(-1) !== "/") { - return folder + "/"; - } else { - return folder; - } - }); - this.file_exclusions = this.file_exclusions.concat(folder_exclusions); - } - if (this.settings.header_exclusions && this.settings.header_exclusions.length > 0) { - this.header_exclusions = this.settings.header_exclusions.split(",").map((header) => { - return header.trim(); - }); - } - if (this.settings.path_only && this.settings.path_only.length > 0) { - this.path_only = this.settings.path_only.split(",").map((path) => { - return path.trim(); - }); - } - this.self_ref_kw_regex = new RegExp(`\\b(${SMART_TRANSLATION[this.settings.language].pronous.join("|")})\\b`, "gi"); - await this.load_failed_files(); - } - async saveSettings(rerender = false) { - await this.saveData(this.settings); - await this.loadSettings(); - if (rerender) { - this.nearest_cache = {}; - await this.make_connections(); - } - } - // check for update - async check_for_update() { - try { - const response = await (0, Obsidian.requestUrl)({ - url: "https://api.github.com/repos/brianpetro/obsidian-smart-connections/releases/latest", - method: "GET", - headers: { - "Content-Type": "application/json" - }, - contentType: "application/json" - }); - const latest_release = JSON.parse(response.text).tag_name; - if (latest_release !== VERSION) { - new Obsidian.Notice(`[Smart Connections] A new version is available! (v${latest_release})`); - this.update_available = true; - this.render_brand("all"); - } - } catch (error) { - console.log(error); - } - } - async render_code_block(contents, container, ctx) { - let nearest; - if (contents.trim().length > 0) { - nearest = await this.api.search(contents); - } else { - console.log(ctx); - const file = this.app.vault.getAbstractFileByPath(ctx.sourcePath); - nearest = await this.find_note_connections(file); - } - if (nearest.length) { - this.update_results(container, nearest); - } - } - async make_connections(selected_text = null) { - let view = this.get_view(); - if (!view) { - await this.open_view(); - view = this.get_view(); - } - await view.render_connections(selected_text); - } - addIcon() { - Obsidian.addIcon("smart-connections", ` - - - - - - `); - } - // open random note - async open_random_note() { - const curr_file = this.app.workspace.getActiveFile(); - const curr_key = md5(curr_file.path); - if (typeof this.nearest_cache[curr_key] === "undefined") { - new Obsidian.Notice("[Smart Connections] No Smart Connections found. Open a note to get Smart Connections."); - return; - } - const rand = Math.floor(Math.random() * this.nearest_cache[curr_key].length / 2); - const random_file = this.nearest_cache[curr_key][rand]; - this.open_note(random_file); - } - async open_view() { - if (this.get_view()) { - console.log("Smart Connections view already open"); - return; - } - this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE); - await this.app.workspace.getRightLeaf(false).setViewState({ - type: SMART_CONNECTIONS_VIEW_TYPE, - active: true - }); - this.app.workspace.revealLeaf( - this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE)[0] - ); - } - // source: https://github.com/obsidianmd/obsidian-releases/blob/master/plugin-review.md#avoid-managing-references-to-custom-views - get_view() { - for (let leaf of this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE)) { - if (leaf.view instanceof SmartConnectionsView) { - return leaf.view; - } - } - } - // open chat view - async open_chat(retries = 0) { - if (!this.embeddings_loaded) { - console.log("embeddings not loaded yet"); - if (retries < 3) { - setTimeout(() => { - this.open_chat(retries + 1); - }, 1e3 * (retries + 1)); - return; - } - console.log("embeddings still not loaded, opening smart view"); - this.open_view(); - return; - } - this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE); - await this.app.workspace.getRightLeaf(false).setViewState({ - type: SMART_CONNECTIONS_CHAT_VIEW_TYPE, - active: true - }); - this.app.workspace.revealLeaf( - this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE)[0] - ); - } - // get embeddings for all files - async get_all_embeddings() { - const files = (await this.app.vault.getFiles()).filter((file) => file instanceof Obsidian.TFile && (file.extension === "md" || file.extension === "canvas")); - const open_files = this.app.workspace.getLeavesOfType("markdown").map((leaf) => leaf.view.file); - const clean_up_log = this.smart_vec_lite.clean_up_embeddings(files); - if (this.settings.log_render) { - this.render_log.total_files = files.length; - this.render_log.deleted_embeddings = clean_up_log.deleted_embeddings; - this.render_log.total_embeddings = clean_up_log.total_embeddings; - } - let batch_promises = []; - for (let i = 0; i < files.length; i++) { - if (files[i].path.indexOf("#") > -1) { - this.log_exclusion("path contains #"); - continue; - } - if (this.smart_vec_lite.mtime_is_current(md5(files[i].path), files[i].stat.mtime)) { - continue; - } - if (this.settings.failed_files.indexOf(files[i].path) > -1) { - if (this.retry_notice_timeout) { - clearTimeout(this.retry_notice_timeout); - this.retry_notice_timeout = null; - } - if (!this.recently_sent_retry_notice) { - new Obsidian.Notice("Smart Connections: Skipping previously failed file, use button in settings to retry"); - this.recently_sent_retry_notice = true; - setTimeout(() => { - this.recently_sent_retry_notice = false; - }, 6e5); - } - continue; - } - let skip = false; - for (let j = 0; j < this.file_exclusions.length; j++) { - if (files[i].path.indexOf(this.file_exclusions[j]) > -1) { - skip = true; - this.log_exclusion(this.file_exclusions[j]); - break; - } - } - if (skip) { - continue; - } - if (open_files.indexOf(files[i]) > -1) { - continue; - } - try { - batch_promises.push(this.get_file_embeddings(files[i], false)); - } catch (error) { - console.log(error); - } - if (batch_promises.length > 3) { - await Promise.all(batch_promises); - batch_promises = []; - } - if (i > 0 && i % 100 === 0) { - await this.save_embeddings_to_file(); - } - } - await Promise.all(batch_promises); - await this.save_embeddings_to_file(); - if (this.render_log.failed_embeddings.length > 0) { - await this.save_failed_embeddings(); - } - } - async save_embeddings_to_file(force = false) { - if (!this.has_new_embeddings) { - return; - } - if (!force) { - if (this.save_timeout) { - clearTimeout(this.save_timeout); - this.save_timeout = null; - } - this.save_timeout = setTimeout(() => { - this.save_embeddings_to_file(true); - if (this.save_timeout) { - clearTimeout(this.save_timeout); - this.save_timeout = null; - } - }, 3e4); - console.log("scheduled save"); - return; - } - try { - await this.smart_vec_lite.save(); - this.has_new_embeddings = false; - } catch (error) { - console.log(error); - new Obsidian.Notice("Smart Connections: " + error.message); - } - } - // save failed embeddings to file from render_log.failed_embeddings - async save_failed_embeddings() { - let failed_embeddings = []; - const failed_embeddings_file_exists = await this.app.vault.adapter.exists(".smart-connections/failed-embeddings.txt"); - if (failed_embeddings_file_exists) { - failed_embeddings = await this.app.vault.adapter.read(".smart-connections/failed-embeddings.txt"); - failed_embeddings = failed_embeddings.split("\r\n"); - } - failed_embeddings = failed_embeddings.concat(this.render_log.failed_embeddings); - failed_embeddings = [...new Set(failed_embeddings)]; - failed_embeddings.sort(); - failed_embeddings = failed_embeddings.join("\r\n"); - await this.app.vault.adapter.write(".smart-connections/failed-embeddings.txt", failed_embeddings); - await this.load_failed_files(); - } - // load failed files from failed-embeddings.txt - async load_failed_files() { - const failed_embeddings_file_exists = await this.app.vault.adapter.exists(".smart-connections/failed-embeddings.txt"); - if (!failed_embeddings_file_exists) { - this.settings.failed_files = []; - console.log("No failed files."); - return; - } - const failed_embeddings = await this.app.vault.adapter.read(".smart-connections/failed-embeddings.txt"); - const failed_embeddings_array = failed_embeddings.split("\r\n"); - const failed_files = failed_embeddings_array.map((embedding) => embedding.split("#")[0]).reduce((unique, item) => unique.includes(item) ? unique : [...unique, item], []); - this.settings.failed_files = failed_files; - } - // retry failed embeddings - async retry_failed_files() { - this.settings.failed_files = []; - const failed_embeddings_file_exists = await this.app.vault.adapter.exists(".smart-connections/failed-embeddings.txt"); - if (failed_embeddings_file_exists) { - await this.app.vault.adapter.remove(".smart-connections/failed-embeddings.txt"); - } - await this.get_all_embeddings(); - } - // add .smart-connections to .gitignore to prevent issues with large, frequently updated embeddings file(s) - async add_to_gitignore() { - if (!await this.app.vault.adapter.exists(".gitignore")) { - return; - } - let gitignore_file = await this.app.vault.adapter.read(".gitignore"); - if (gitignore_file.indexOf(".smart-connections") < 0) { - let add_to_gitignore = "\n\n# Ignore Smart Connections folder because embeddings file is large and updated frequently"; - add_to_gitignore += "\n.smart-connections"; - await this.app.vault.adapter.write(".gitignore", gitignore_file + add_to_gitignore); - console.log("added .smart-connections to .gitignore"); - } - } - // force refresh embeddings file but first rename existing embeddings file to .smart-connections/embeddings-YYYY-MM-DD.json - async force_refresh_embeddings_file() { - new Obsidian.Notice("Smart Connections: embeddings file Force Refreshed, making new connections..."); - await this.smart_vec_lite.force_refresh(); - await this.get_all_embeddings(); - this.output_render_log(); - new Obsidian.Notice("Smart Connections: embeddings file Force Refreshed, new connections made."); - } - // get embeddings for embed_input - async get_file_embeddings(curr_file, save = true) { - let req_batch = []; - let blocks = []; - const curr_file_key = md5(curr_file.path); - let file_embed_input = curr_file.path.replace(".md", ""); - file_embed_input = file_embed_input.replace(/\//g, " > "); - let path_only = false; - for (let j = 0; j < this.path_only.length; j++) { - if (curr_file.path.indexOf(this.path_only[j]) > -1) { - path_only = true; - console.log("title only file with matcher: " + this.path_only[j]); - break; - } - } - if (path_only) { - req_batch.push([curr_file_key, file_embed_input, { - mtime: curr_file.stat.mtime, - path: curr_file.path - }]); - await this.get_embeddings_batch(req_batch); - return; - } - if (curr_file.extension === "canvas") { - const canvas_contents = await this.app.vault.cachedRead(curr_file); - if (typeof canvas_contents === "string" && canvas_contents.indexOf("nodes") > -1) { - const canvas_json = JSON.parse(canvas_contents); - for (let j = 0; j < canvas_json.nodes.length; j++) { - if (canvas_json.nodes[j].text) { - file_embed_input += "\n" + canvas_json.nodes[j].text; - } - if (canvas_json.nodes[j].file) { - file_embed_input += "\nLink: " + canvas_json.nodes[j].file; - } - } - } - req_batch.push([curr_file_key, file_embed_input, { - mtime: curr_file.stat.mtime, - path: curr_file.path - }]); - await this.get_embeddings_batch(req_batch); - return; - } - const note_contents = await this.app.vault.cachedRead(curr_file); - let processed_since_last_save = 0; - const note_sections = this.block_parser(note_contents, curr_file.path); - if (note_sections.length > 1) { - for (let j = 0; j < note_sections.length; j++) { - const block_embed_input = note_sections[j].text; - const block_key = md5(note_sections[j].path); - blocks.push(block_key); - if (this.smart_vec_lite.get_size(block_key) === block_embed_input.length) { - continue; - } - if (this.smart_vec_lite.mtime_is_current(block_key, curr_file.stat.mtime)) { - continue; - } - const block_hash = md5(block_embed_input.trim()); - if (this.smart_vec_lite.get_hash(block_key) === block_hash) { - continue; - } - req_batch.push([block_key, block_embed_input, { - // oldmtime: curr_file.stat.mtime, - // get current datetime as unix timestamp - mtime: Date.now(), - hash: block_hash, - parent: curr_file_key, - path: note_sections[j].path, - size: block_embed_input.length - }]); - if (req_batch.length > 9) { - await this.get_embeddings_batch(req_batch); - processed_since_last_save += req_batch.length; - if (processed_since_last_save >= 30) { - await this.save_embeddings_to_file(); - processed_since_last_save = 0; - } - req_batch = []; - } - } - } - if (req_batch.length > 0) { - await this.get_embeddings_batch(req_batch); - req_batch = []; - processed_since_last_save += req_batch.length; - } - file_embed_input += `: -`; - if (note_contents.length < MAX_EMBED_STRING_LENGTH) { - file_embed_input += note_contents; - } else { - const note_meta_cache = this.app.metadataCache.getFileCache(curr_file); - if (typeof note_meta_cache.headings === "undefined") { - file_embed_input += note_contents.substring(0, MAX_EMBED_STRING_LENGTH); - } else { - let note_headings = ""; - for (let j = 0; j < note_meta_cache.headings.length; j++) { - const heading_level = note_meta_cache.headings[j].level; - const heading_text = note_meta_cache.headings[j].heading; - let md_heading = ""; - for (let k = 0; k < heading_level; k++) { - md_heading += "#"; - } - note_headings += `${md_heading} ${heading_text} -`; - } - file_embed_input += note_headings; - if (file_embed_input.length > MAX_EMBED_STRING_LENGTH) { - file_embed_input = file_embed_input.substring(0, MAX_EMBED_STRING_LENGTH); - } - } - } - const file_hash = md5(file_embed_input.trim()); - const existing_hash = this.smart_vec_lite.get_hash(curr_file_key); - if (existing_hash && file_hash === existing_hash) { - this.update_render_log(blocks, file_embed_input); - return; - } - ; - const existing_blocks = this.smart_vec_lite.get_children(curr_file_key); - let existing_has_all_blocks = true; - if (existing_blocks && Array.isArray(existing_blocks) && blocks.length > 0) { - for (let j = 0; j < blocks.length; j++) { - if (existing_blocks.indexOf(blocks[j]) === -1) { - existing_has_all_blocks = false; - break; - } - } - } - if (existing_has_all_blocks) { - const curr_file_size = curr_file.stat.size; - const prev_file_size = this.smart_vec_lite.get_size(curr_file_key); - if (prev_file_size) { - const file_delta_pct = Math.round(Math.abs(curr_file_size - prev_file_size) / curr_file_size * 100); - if (file_delta_pct < 10) { - this.render_log.skipped_low_delta[curr_file.name] = file_delta_pct + "%"; - this.update_render_log(blocks, file_embed_input); - return; - } - } - } - let meta = { - mtime: curr_file.stat.mtime, - hash: file_hash, - path: curr_file.path, - size: curr_file.stat.size, - children: blocks - }; - req_batch.push([curr_file_key, file_embed_input, meta]); - await this.get_embeddings_batch(req_batch); - if (save) { - await this.save_embeddings_to_file(); - } - } - update_render_log(blocks, file_embed_input) { - if (blocks.length > 0) { - this.render_log.tokens_saved_by_cache += file_embed_input.length / 2; - } else { - this.render_log.tokens_saved_by_cache += file_embed_input.length / 4; - } - } - async get_embeddings_batch(req_batch) { - console.log("get_embeddings_batch"); - if (req_batch.length === 0) - return; - const embed_inputs = req_batch.map((req) => req[1]); - const requestResults = await this.request_embedding_from_input(embed_inputs); - if (!requestResults) { - console.log("failed embedding batch"); - this.render_log.failed_embeddings = [...this.render_log.failed_embeddings, ...req_batch.map((req) => req[2].path)]; - return; - } - if (requestResults) { - this.has_new_embeddings = true; - if (this.settings.log_render) { - if (this.settings.log_render_files) { - this.render_log.files = [...this.render_log.files, ...req_batch.map((req) => req[2].path)]; - } - this.render_log.new_embeddings += req_batch.length; - this.render_log.token_usage += requestResults.usage.total_tokens; - } - for (let i = 0; i < requestResults.data.length; i++) { - const vec = requestResults.data[i].embedding; - const index = requestResults.data[i].index; - if (vec) { - const key = req_batch[index][0]; - const meta = req_batch[index][2]; - this.smart_vec_lite.save_embedding(key, vec, meta); - } - } - } - } - async request_embedding_from_input(embed_input, retries = 0) { - if (embed_input.length === 0) { - console.log("embed_input is empty"); - return null; - } - const usedParams = { - model: "text-embedding-ada-002", - input: embed_input - }; - const reqParams = { - url: `https://api.openai.com/v1/embeddings`, - method: "POST", - body: JSON.stringify(usedParams), - headers: { - "Content-Type": "application/json", - "Authorization": `Bearer ${this.settings.api_key}` - } - }; - let resp; - try { - resp = await (0, Obsidian.request)(reqParams); - return JSON.parse(resp); - } catch (error) { - if (error.status === 429 && retries < 3) { - retries++; - const backoff = Math.pow(retries, 2); - console.log(`retrying request (429) in ${backoff} seconds...`); - await new Promise((r) => setTimeout(r, 1e3 * backoff)); - return await this.request_embedding_from_input(embed_input, retries); - } - console.log(resp); - console.log(error); - return null; - } - } - async test_api_key() { - const embed_input = "This is a test of the OpenAI API."; - const resp = await this.request_embedding_from_input(embed_input); - if (resp && resp.usage) { - console.log("API key is valid"); - return true; - } else { - console.log("API key is invalid"); - return false; - } - } - output_render_log() { - if (this.settings.log_render) { - if (this.render_log.new_embeddings === 0) { - return; - } else { - console.log(JSON.stringify(this.render_log, null, 2)); - } - } - this.render_log = {}; - this.render_log.deleted_embeddings = 0; - this.render_log.exclusions_logs = {}; - this.render_log.failed_embeddings = []; - this.render_log.files = []; - this.render_log.new_embeddings = 0; - this.render_log.skipped_low_delta = {}; - this.render_log.token_usage = 0; - this.render_log.tokens_saved_by_cache = 0; - } - // find connections by most similar to current note by cosine similarity - async find_note_connections(current_note = null) { - const curr_key = md5(current_note.path); - let nearest = []; - if (this.nearest_cache[curr_key]) { - nearest = this.nearest_cache[curr_key]; - } else { - for (let j = 0; j < this.file_exclusions.length; j++) { - if (current_note.path.indexOf(this.file_exclusions[j]) > -1) { - this.log_exclusion(this.file_exclusions[j]); - return "excluded"; - } - } - setTimeout(() => { - this.get_all_embeddings(); - }, 3e3); - if (this.smart_vec_lite.mtime_is_current(curr_key, current_note.stat.mtime)) { - } else { - await this.get_file_embeddings(current_note); - } - const vec = this.smart_vec_lite.get_vec(curr_key); - if (!vec) { - return "Error getting embeddings for: " + current_note.path; - } - nearest = this.smart_vec_lite.nearest(vec, { - skip_key: curr_key, - skip_sections: this.settings.skip_sections - }); - this.nearest_cache[curr_key] = nearest; - } - return nearest; - } - // create render_log object of exlusions with number of times skipped as value - log_exclusion(exclusion) { - this.render_log.exclusions_logs[exclusion] = (this.render_log.exclusions_logs[exclusion] || 0) + 1; - } - block_parser(markdown, file_path) { - if (this.settings.skip_sections) { - return []; - } - const lines = markdown.split("\n"); - let blocks = []; - let currentHeaders = []; - const file_breadcrumbs = file_path.replace(".md", "").replace(/\//g, " > "); - let block = ""; - let block_headings = ""; - let block_path = file_path; - let last_heading_line = 0; - let i = 0; - let block_headings_list = []; - for (i = 0; i < lines.length; i++) { - const line = lines[i]; - if (!line.startsWith("#") || ["#", " "].indexOf(line[1]) < 0) { - if (line === "") - continue; - if (["- ", "- [ ] "].indexOf(line) > -1) - continue; - if (currentHeaders.length === 0) - continue; - block += "\n" + line; - continue; - } - last_heading_line = i; - if (i > 0 && last_heading_line !== i - 1 && block.indexOf("\n") > -1 && this.validate_headings(block_headings)) { - output_block(); - } - const level = line.split("#").length - 1; - currentHeaders = currentHeaders.filter((header) => header.level < level); - currentHeaders.push({ header: line.replace(/#/g, "").trim(), level }); - block = file_breadcrumbs; - block += ": " + currentHeaders.map((header) => header.header).join(" > "); - block_headings = "#" + currentHeaders.map((header) => header.header).join("#"); - if (block_headings_list.indexOf(block_headings) > -1) { - let count = 1; - while (block_headings_list.indexOf(`${block_headings}{${count}}`) > -1) { - count++; - } - block_headings = `${block_headings}{${count}}`; - } - block_headings_list.push(block_headings); - block_path = file_path + block_headings; - } - if (last_heading_line !== i - 1 && block.indexOf("\n") > -1 && this.validate_headings(block_headings)) - output_block(); - blocks = blocks.filter((b) => b.length > 50); - return blocks; - function output_block() { - const breadcrumbs_length = block.indexOf("\n") + 1; - const block_length = block.length - breadcrumbs_length; - if (block.length > MAX_EMBED_STRING_LENGTH) { - block = block.substring(0, MAX_EMBED_STRING_LENGTH); - } - blocks.push({ text: block.trim(), path: block_path, length: block_length }); - } - } - // reverse-retrieve block given path - async block_retriever(path, limits = {}) { - limits = { - lines: null, - chars_per_line: null, - max_chars: null, - ...limits - }; - if (path.indexOf("#") < 0) { - console.log("not a block path: " + path); - return false; - } - let block = []; - let block_headings = path.split("#").slice(1); - let heading_occurrence = 0; - if (block_headings[block_headings.length - 1].indexOf("{") > -1) { - heading_occurrence = parseInt(block_headings[block_headings.length - 1].split("{")[1].replace("}", "")); - block_headings[block_headings.length - 1] = block_headings[block_headings.length - 1].split("{")[0]; - } - let currentHeaders = []; - let occurrence_count = 0; - let begin_line = 0; - let i = 0; - const file_path = path.split("#")[0]; - const file = this.app.vault.getAbstractFileByPath(file_path); - if (!(file instanceof Obsidian.TFile)) { - console.log("not a file: " + file_path); - return false; - } - const file_contents = await this.app.vault.cachedRead(file); - const lines = file_contents.split("\n"); - let is_code = false; - for (i = 0; i < lines.length; i++) { - const line = lines[i]; - if (line.indexOf("```") === 0) { - is_code = !is_code; - } - if (is_code) { - continue; - } - if (["- ", "- [ ] "].indexOf(line) > -1) - continue; - if (!line.startsWith("#") || ["#", " "].indexOf(line[1]) < 0) { - continue; - } - const heading_text = line.replace(/#/g, "").trim(); - const heading_index = block_headings.indexOf(heading_text); - if (heading_index < 0) - continue; - if (currentHeaders.length !== heading_index) - continue; - currentHeaders.push(heading_text); - if (currentHeaders.length === block_headings.length) { - if (heading_occurrence === 0) { - begin_line = i + 1; - break; - } - if (occurrence_count === heading_occurrence) { - begin_line = i + 1; - break; - } - occurrence_count++; - currentHeaders.pop(); - continue; - } - } - if (begin_line === 0) - return false; - is_code = false; - let char_count = 0; - for (i = begin_line; i < lines.length; i++) { - if (typeof line_limit === "number" && block.length > line_limit) { - block.push("..."); - break; - } - let line = lines[i]; - if (line.indexOf("#") === 0 && ["#", " "].indexOf(line[1]) !== -1) { - break; - } - if (limits.max_chars && char_count > limits.max_chars) { - block.push("..."); - break; - } - if (limits.max_chars && line.length + char_count > limits.max_chars) { - const max_new_chars = limits.max_chars - char_count; - line = line.slice(0, max_new_chars) + "..."; - break; - } - if (line.length === 0) - continue; - if (limits.chars_per_line && line.length > limits.chars_per_line) { - line = line.slice(0, limits.chars_per_line) + "..."; - } - if (line.startsWith("```")) { - is_code = !is_code; - continue; - } - if (is_code) { - line = " " + line; - } - block.push(line); - char_count += line.length; - } - if (is_code) { - block.push("```"); - } - return block.join("\n").trim(); - } - // retrieve a file from the vault - async file_retriever(link, limits = {}) { - limits = { - lines: null, - max_chars: null, - chars_per_line: null, - ...limits - }; - const this_file = this.app.vault.getAbstractFileByPath(link); - if (!(this_file instanceof Obsidian.TAbstractFile)) - return false; - const file_content = await this.app.vault.cachedRead(this_file); - const file_lines = file_content.split("\n"); - let first_ten_lines = []; - let is_code = false; - let char_accum = 0; - const line_limit2 = limits.lines || file_lines.length; - for (let i = 0; first_ten_lines.length < line_limit2; i++) { - let line = file_lines[i]; - if (typeof line === "undefined") - break; - if (line.length === 0) - continue; - if (limits.chars_per_line && line.length > limits.chars_per_line) { - line = line.slice(0, limits.chars_per_line) + "..."; - } - if (line === "---") - continue; - if (["- ", "- [ ] "].indexOf(line) > -1) - continue; - if (line.indexOf("```") === 0) { - is_code = !is_code; - continue; - } - if (limits.max_chars && char_accum > limits.max_chars) { - first_ten_lines.push("..."); - break; - } - if (is_code) { - line = " " + line; - } - if (line_is_heading(line)) { - if (first_ten_lines.length > 0 && line_is_heading(first_ten_lines[first_ten_lines.length - 1])) { - first_ten_lines.pop(); - } - } - first_ten_lines.push(line); - char_accum += line.length; - } - for (let i = 0; i < first_ten_lines.length; i++) { - if (line_is_heading(first_ten_lines[i])) { - if (i === first_ten_lines.length - 1) { - first_ten_lines.pop(); - break; - } - first_ten_lines[i] = first_ten_lines[i].replace(/#+/, ""); - first_ten_lines[i] = ` -${first_ten_lines[i]}:`; - } - } - first_ten_lines = first_ten_lines.join("\n"); - return first_ten_lines; - } - // iterate through blocks and skip if block_headings contains this.header_exclusions - validate_headings(block_headings) { - let valid = true; - if (this.header_exclusions.length > 0) { - for (let k = 0; k < this.header_exclusions.length; k++) { - if (block_headings.indexOf(this.header_exclusions[k]) > -1) { - valid = false; - this.log_exclusion("heading: " + this.header_exclusions[k]); - break; - } - } - } - return valid; - } - // render "Smart Connections" text fixed in the bottom right corner - render_brand(container, location = "default") { - if (container === "all") { - const locations = Object.keys(this.sc_branding); - for (let i = 0; i < locations.length; i++) { - this.render_brand(this.sc_branding[locations[i]], locations[i]); - } - return; - } - this.sc_branding[location] = container; - if (this.sc_branding[location].querySelector(".sc-brand")) { - this.sc_branding[location].querySelector(".sc-brand").remove(); - } - const brand_container = this.sc_branding[location].createEl("div", { cls: "sc-brand" }); - Obsidian.setIcon(brand_container, "smart-connections"); - const brand_p = brand_container.createEl("p"); - let text = "Smart Connections"; - let attr = {}; - if (this.update_available) { - text = "Update Available"; - attr = { - style: "font-weight: 700;" - }; - } - brand_p.createEl("a", { - cls: "", - text, - href: "https://github.com/brianpetro/obsidian-smart-connections/discussions", - target: "_blank", - attr - }); - } - // create list of nearest notes - async update_results(container, nearest) { - let list; - if (container.children.length > 1 && container.children[1].classList.contains("sc-list")) { - list = container.children[1]; - } - if (list) { - list.empty(); - } else { - list = container.createEl("div", { cls: "sc-list" }); - } - let search_result_class = "search-result"; - if (!this.settings.expanded_view) - search_result_class += " sc-collapsed"; - if (!this.settings.group_nearest_by_file) { - for (let i = 0; i < nearest.length; i++) { - if (typeof nearest[i].link === "object") { - const item2 = list.createEl("div", { cls: "search-result" }); - const link2 = item2.createEl("a", { - cls: "search-result-file-title is-clickable", - href: nearest[i].link.path, - title: nearest[i].link.title - }); - link2.innerHTML = this.render_external_link_elm(nearest[i].link); - item2.setAttr("draggable", "true"); - continue; - } - let file_link_text; - const file_similarity_pct = Math.round(nearest[i].similarity * 100) + "%"; - if (this.settings.show_full_path) { - const pcs = nearest[i].link.split("/"); - file_link_text = pcs[pcs.length - 1]; - const path = pcs.slice(0, pcs.length - 1).join("/"); - file_link_text = `${file_similarity_pct} | ${path} | ${file_link_text}`; - } else { - file_link_text = "" + file_similarity_pct + " | " + nearest[i].link.split("/").pop() + ""; - } - if (!this.renderable_file_type(nearest[i].link)) { - const item2 = list.createEl("div", { cls: "search-result" }); - const link2 = item2.createEl("a", { - cls: "search-result-file-title is-clickable", - href: nearest[i].link - }); - link2.innerHTML = file_link_text; - item2.setAttr("draggable", "true"); - this.add_link_listeners(link2, nearest[i], item2); - continue; - } - file_link_text = file_link_text.replace(".md", "").replace(/#/g, " > "); - const item = list.createEl("div", { cls: search_result_class }); - const toggle = item.createEl("span", { cls: "is-clickable" }); - Obsidian.setIcon(toggle, "right-triangle"); - const link = toggle.createEl("a", { - cls: "search-result-file-title", - title: nearest[i].link - }); - link.innerHTML = file_link_text; - this.add_link_listeners(link, nearest[i], item); - toggle.addEventListener("click", (event) => { - let parent = event.target.parentElement; - while (!parent.classList.contains("search-result")) { - parent = parent.parentElement; - } - parent.classList.toggle("sc-collapsed"); - }); - const contents = item.createEl("ul", { cls: "" }); - const contents_container = contents.createEl("li", { - cls: "search-result-file-title is-clickable", - title: nearest[i].link - }); - if (nearest[i].link.indexOf("#") > -1) { - Obsidian.MarkdownRenderer.renderMarkdown(await this.block_retriever(nearest[i].link, { lines: 10, max_chars: 1e3 }), contents_container, nearest[i].link, new Obsidian.Component()); - } else { - const first_ten_lines = await this.file_retriever(nearest[i].link, { lines: 10, max_chars: 1e3 }); - if (!first_ten_lines) - continue; - Obsidian.MarkdownRenderer.renderMarkdown(first_ten_lines, contents_container, nearest[i].link, new Obsidian.Component()); - } - this.add_link_listeners(contents, nearest[i], item); - } - this.render_brand(container, "block"); - return; - } - const nearest_by_file = {}; - for (let i = 0; i < nearest.length; i++) { - const curr = nearest[i]; - const link = curr.link; - if (typeof link === "object") { - nearest_by_file[link.path] = [curr]; - continue; - } - if (link.indexOf("#") > -1) { - const file_path = link.split("#")[0]; - if (!nearest_by_file[file_path]) { - nearest_by_file[file_path] = []; - } - nearest_by_file[file_path].push(nearest[i]); - } else { - if (!nearest_by_file[link]) { - nearest_by_file[link] = []; - } - nearest_by_file[link].unshift(nearest[i]); - } - } - const keys = Object.keys(nearest_by_file); - for (let i = 0; i < keys.length; i++) { - const file = nearest_by_file[keys[i]]; - if (typeof file[0].link === "object") { - const curr = file[0]; - const meta = curr.link; - if (meta.path.startsWith("http")) { - const item2 = list.createEl("div", { cls: "search-result" }); - const link = item2.createEl("a", { - cls: "search-result-file-title is-clickable", - href: meta.path, - title: meta.title - }); - link.innerHTML = this.render_external_link_elm(meta); - item2.setAttr("draggable", "true"); - continue; - } - } - let file_link_text; - const file_similarity_pct = Math.round(file[0].similarity * 100) + "%"; - if (this.settings.show_full_path) { - const pcs = file[0].link.split("/"); - file_link_text = pcs[pcs.length - 1]; - const path = pcs.slice(0, pcs.length - 1).join("/"); - file_link_text = `${path} | ${file_similarity_pct}
${file_link_text}`; - } else { - file_link_text = file[0].link.split("/").pop(); - file_link_text += " | " + file_similarity_pct; - } - if (!this.renderable_file_type(file[0].link)) { - const item2 = list.createEl("div", { cls: "search-result" }); - const file_link2 = item2.createEl("a", { - cls: "search-result-file-title is-clickable", - title: file[0].link - }); - file_link2.innerHTML = file_link_text; - this.add_link_listeners(file_link2, file[0], item2); - continue; - } - file_link_text = file_link_text.replace(".md", "").replace(/#/g, " > "); - const item = list.createEl("div", { cls: search_result_class }); - const toggle = item.createEl("span", { cls: "is-clickable" }); - Obsidian.setIcon(toggle, "right-triangle"); - const file_link = toggle.createEl("a", { - cls: "search-result-file-title", - title: file[0].link - }); - file_link.innerHTML = file_link_text; - this.add_link_listeners(file_link, file[0], toggle); - toggle.addEventListener("click", (event) => { - let parent = event.target; - while (!parent.classList.contains("search-result")) { - parent = parent.parentElement; - } - parent.classList.toggle("sc-collapsed"); - }); - const file_link_list = item.createEl("ul"); - for (let j = 0; j < file.length; j++) { - if (file[j].link.indexOf("#") > -1) { - const block = file[j]; - const block_link = file_link_list.createEl("li", { - cls: "search-result-file-title is-clickable", - title: block.link - }); - if (file.length > 1) { - const block_context = this.render_block_context(block); - const block_similarity_pct = Math.round(block.similarity * 100) + "%"; - block_link.innerHTML = `${block_context} | ${block_similarity_pct}`; - } - const block_container = block_link.createEl("div"); - Obsidian.MarkdownRenderer.renderMarkdown(await this.block_retriever(block.link, { lines: 10, max_chars: 1e3 }), block_container, block.link, new Obsidian.Component()); - this.add_link_listeners(block_link, block, file_link_list); - } else { - const file_link_list2 = item.createEl("ul"); - const block_link = file_link_list2.createEl("li", { - cls: "search-result-file-title is-clickable", - title: file[0].link - }); - const block_container = block_link.createEl("div"); - let first_ten_lines = await this.file_retriever(file[0].link, { lines: 10, max_chars: 1e3 }); - if (!first_ten_lines) - continue; - Obsidian.MarkdownRenderer.renderMarkdown(first_ten_lines, block_container, file[0].link, new Obsidian.Component()); - this.add_link_listeners(block_link, file[0], file_link_list2); - } - } - } - this.render_brand(container, "file"); - } - add_link_listeners(item, curr, list) { - item.addEventListener("click", async (event) => { - await this.open_note(curr, event); - }); - item.setAttr("draggable", "true"); - item.addEventListener("dragstart", (event) => { - const dragManager = this.app.dragManager; - const file_path = curr.link.split("#")[0]; - const file = this.app.metadataCache.getFirstLinkpathDest(file_path, ""); - const dragData = dragManager.dragFile(event, file); - dragManager.onDragStart(event, dragData); - }); - if (curr.link.indexOf("{") > -1) - return; - item.addEventListener("mouseover", (event) => { - this.app.workspace.trigger("hover-link", { - event, - source: SMART_CONNECTIONS_VIEW_TYPE, - hoverParent: list, - targetEl: item, - linktext: curr.link - }); - }); - } - // get target file from link path - // if sub-section is linked, open file and scroll to sub-section - async open_note(curr, event = null) { - let targetFile; - let heading; - if (curr.link.indexOf("#") > -1) { - targetFile = this.app.metadataCache.getFirstLinkpathDest(curr.link.split("#")[0], ""); - const target_file_cache = this.app.metadataCache.getFileCache(targetFile); - let heading_text = curr.link.split("#").pop(); - let occurence = 0; - if (heading_text.indexOf("{") > -1) { - occurence = parseInt(heading_text.split("{")[1].split("}")[0]); - heading_text = heading_text.split("{")[0]; - } - const headings = target_file_cache.headings; - for (let i = 0; i < headings.length; i++) { - if (headings[i].heading === heading_text) { - if (occurence === 0) { - heading = headings[i]; - break; - } - occurence--; - } - } - } else { - targetFile = this.app.metadataCache.getFirstLinkpathDest(curr.link, ""); - } - let leaf; - if (event) { - const mod = Obsidian.Keymap.isModEvent(event); - leaf = this.app.workspace.getLeaf(mod); - } else { - leaf = this.app.workspace.getMostRecentLeaf(); - } - await leaf.openFile(targetFile); - if (heading) { - let { editor } = leaf.view; - const pos = { line: heading.position.start.line, ch: 0 }; - editor.setCursor(pos); - editor.scrollIntoView({ to: pos, from: pos }, true); - } - } - render_block_context(block) { - const block_headings = block.link.split(".md")[1].split("#"); - let block_context = ""; - for (let i = block_headings.length - 1; i >= 0; i--) { - if (block_context.length > 0) { - block_context = ` > ${block_context}`; - } - block_context = block_headings[i] + block_context; - if (block_context.length > 100) { - break; - } - } - if (block_context.startsWith(" > ")) { - block_context = block_context.slice(3); - } - return block_context; - } - renderable_file_type(link) { - return link.indexOf(".md") !== -1 && link.indexOf(".excalidraw") === -1; - } - render_external_link_elm(meta) { - if (meta.source) { - if (meta.source === "Gmail") - meta.source = "\u{1F4E7} Gmail"; - return `${meta.source}
${meta.title}`; - } - let domain = meta.path.replace(/(^\w+:|^)\/\//, ""); - domain = domain.split("/")[0]; - return `\u{1F310} ${domain}
${meta.title}`; - } - // get all folders - async get_all_folders() { - if (!this.folders || this.folders.length === 0) { - this.folders = await this.get_folders(); - } - return this.folders; - } - // get folders, traverse non-hidden sub-folders - async get_folders(path = "/") { - let folders = (await this.app.vault.adapter.list(path)).folders; - let folder_list = []; - for (let i = 0; i < folders.length; i++) { - if (folders[i].startsWith(".")) - continue; - folder_list.push(folders[i]); - folder_list = folder_list.concat(await this.get_folders(folders[i] + "/")); - } - return folder_list; - } - async sync_notes() { - if (!this.settings.license_key) { - new Obsidian.Notice("Smart Connections: Supporter license key is required to sync notes to the ChatGPT Plugin server."); - return; - } - console.log("syncing notes"); - const files = this.app.vault.getMarkdownFiles().filter((file) => { - for (let i = 0; i < this.file_exclusions.length; i++) { - if (file.path.indexOf(this.file_exclusions[i]) > -1) { - return false; - } - } - return true; - }); - const notes = await this.build_notes_object(files); - console.log("object built"); - await this.app.vault.adapter.write(".smart-connections/notes.json", JSON.stringify(notes, null, 2)); - console.log("notes saved"); - console.log(this.settings.license_key); - const response = await (0, Obsidian.requestUrl)({ - url: "https://sync.smartconnections.app/sync", - method: "POST", - headers: { - "Content-Type": "application/json" - }, - contentType: "application/json", - body: JSON.stringify({ - license_key: this.settings.license_key, - notes - }) - }); - console.log(response); - } - async build_notes_object(files) { - let output = {}; - for (let i = 0; i < files.length; i++) { - let file = files[i]; - let parts = file.path.split("/"); - let current = output; - for (let ii = 0; ii < parts.length; ii++) { - let part = parts[ii]; - if (ii === parts.length - 1) { - current[part] = await this.app.vault.cachedRead(file); - } else { - if (!current[part]) { - current[part] = {}; - } - current = current[part]; - } - } - } - return output; - } -}; -var SMART_CONNECTIONS_VIEW_TYPE = "smart-connections-view"; -var SmartConnectionsView = class extends Obsidian.ItemView { - constructor(leaf, plugin) { - super(leaf); - this.plugin = plugin; - this.nearest = null; - this.load_wait = null; - } - getViewType() { - return SMART_CONNECTIONS_VIEW_TYPE; - } - getDisplayText() { - return "Smart Connections Files"; - } - getIcon() { - return "smart-connections"; - } - set_message(message) { - const container = this.containerEl.children[1]; - container.empty(); - this.initiate_top_bar(container); - if (Array.isArray(message)) { - for (let i = 0; i < message.length; i++) { - container.createEl("p", { cls: "sc_message", text: message[i] }); - } - } else { - container.createEl("p", { cls: "sc_message", text: message }); - } - } - render_link_text(link, show_full_path = false) { - if (!show_full_path) { - link = link.split("/").pop(); - } - if (link.indexOf("#") > -1) { - link = link.split(".md"); - link[0] = `${link[0]}
`; - link = link.join(""); - link = link.replace(/\#/g, " \xBB "); - } else { - link = link.replace(".md", ""); - } - return link; - } - set_nearest(nearest, nearest_context = null, results_only = false) { - const container = this.containerEl.children[1]; - if (!results_only) { - container.empty(); - this.initiate_top_bar(container, nearest_context); - } - this.plugin.update_results(container, nearest); - } - initiate_top_bar(container, nearest_context = null) { - let top_bar; - if (container.children.length > 0 && container.children[0].classList.contains("sc-top-bar")) { - top_bar = container.children[0]; - top_bar.empty(); - } else { - top_bar = container.createEl("div", { cls: "sc-top-bar" }); - } - if (nearest_context) { - top_bar.createEl("p", { cls: "sc-context", text: nearest_context }); - } - const chat_button = top_bar.createEl("button", { cls: "sc-chat-button" }); - Obsidian.setIcon(chat_button, "message-square"); - chat_button.addEventListener("click", () => { - this.plugin.open_chat(); - }); - const search_button = top_bar.createEl("button", { cls: "sc-search-button" }); - Obsidian.setIcon(search_button, "search"); - search_button.addEventListener("click", () => { - top_bar.empty(); - const search_container = top_bar.createEl("div", { cls: "search-input-container" }); - const input = search_container.createEl("input", { - cls: "sc-search-input", - type: "search", - placeholder: "Type to start search..." - }); - input.focus(); - input.addEventListener("keydown", (event) => { - if (event.key === "Escape") { - this.clear_auto_searcher(); - this.initiate_top_bar(container, nearest_context); - } - }); - input.addEventListener("keyup", (event) => { - this.clear_auto_searcher(); - const search_term = input.value; - if (event.key === "Enter" && search_term !== "") { - this.search(search_term); - } else if (search_term !== "") { - clearTimeout(this.search_timeout); - this.search_timeout = setTimeout(() => { - this.search(search_term, true); - }, 700); - } - }); - }); - } - // render buttons: "create" and "retry" for loading embeddings.json file - render_embeddings_buttons() { - const container = this.containerEl.children[1]; - container.empty(); - container.createEl("h2", { cls: "scHeading", text: "Embeddings file not found" }); - const button_div = container.createEl("div", { cls: "scButtonDiv" }); - const create_button = button_div.createEl("button", { cls: "scButton", text: "Create embeddings.json" }); - button_div.createEl("p", { cls: "scButtonNote", text: "Warning: Creating embeddings.json file will trigger bulk embedding and may take a while" }); - const retry_button = button_div.createEl("button", { cls: "scButton", text: "Retry" }); - button_div.createEl("p", { cls: "scButtonNote", text: "If embeddings.json file already exists, click 'Retry' to load it" }); - create_button.addEventListener("click", async (event) => { - await this.plugin.smart_vec_lite.init_embeddings_file(); - await this.render_connections(); - }); - retry_button.addEventListener("click", async (event) => { - console.log("retrying to load embeddings.json file"); - await this.plugin.init_vecs(); - await this.render_connections(); - }); - } - async onOpen() { - const container = this.containerEl.children[1]; - container.empty(); - container.createEl("p", { cls: "scPlaceholder", text: "Open a note to find connections." }); - this.plugin.registerEvent(this.app.workspace.on("file-open", (file) => { - if (!file) { - return; - } - if (SUPPORTED_FILE_TYPES.indexOf(file.extension) === -1) { - return this.set_message([ - "File: " + file.name, - "Unsupported file type (Supported: " + SUPPORTED_FILE_TYPES.join(", ") + ")" - ]); - } - if (this.load_wait) { - clearTimeout(this.load_wait); - } - this.load_wait = setTimeout(() => { - this.render_connections(file); - this.load_wait = null; - }, 1e3); - })); - this.app.workspace.registerHoverLinkSource(SMART_CONNECTIONS_VIEW_TYPE, { - display: "Smart Connections Files", - defaultMod: true - }); - this.app.workspace.registerHoverLinkSource(SMART_CONNECTIONS_CHAT_VIEW_TYPE, { - display: "Smart Chat Links", - defaultMod: true - }); - this.app.workspace.onLayoutReady(this.initialize.bind(this)); - } - async initialize() { - this.set_message("Loading embeddings file..."); - const vecs_intiated = await this.plugin.init_vecs(); - if (vecs_intiated) { - this.set_message("Embeddings file loaded."); - await this.render_connections(); - } else { - this.render_embeddings_buttons(); - } - this.api = new SmartConnectionsViewApi(this.app, this.plugin, this); - (window["SmartConnectionsViewApi"] = this.api) && this.register(() => delete window["SmartConnectionsViewApi"]); - } - async onClose() { - console.log("closing smart connections view"); - this.app.workspace.unregisterHoverLinkSource(SMART_CONNECTIONS_VIEW_TYPE); - this.plugin.view = null; - } - async render_connections(context = null) { - console.log("rendering connections"); - if (!this.plugin.settings.api_key) { - this.set_message("An OpenAI API key is required to make Smart Connections"); - return; - } - if (!this.plugin.embeddings_loaded) { - await this.plugin.init_vecs(); - } - if (!this.plugin.embeddings_loaded) { - console.log("embeddings files still not loaded or yet to be created"); - this.render_embeddings_buttons(); - return; - } - this.set_message("Making Smart Connections..."); - if (typeof context === "string") { - const highlighted_text = context; - await this.search(highlighted_text); - return; - } - this.nearest = null; - this.interval_count = 0; - this.rendering = false; - this.file = context; - if (this.interval) { - clearInterval(this.interval); - this.interval = null; - } - this.interval = setInterval(() => { - if (!this.rendering) { - if (this.file instanceof Obsidian.TFile) { - this.rendering = true; - this.render_note_connections(this.file); - } else { - this.file = this.app.workspace.getActiveFile(); - if (!this.file && this.count > 1) { - clearInterval(this.interval); - this.set_message("No active file"); - return; - } - } - } else { - if (this.nearest) { - clearInterval(this.interval); - if (typeof this.nearest === "string") { - this.set_message(this.nearest); - } else { - this.set_nearest(this.nearest, "File: " + this.file.name); - } - if (this.plugin.render_log.failed_embeddings.length > 0) { - this.plugin.save_failed_embeddings(); - } - this.plugin.output_render_log(); - return; - } else { - this.interval_count++; - this.set_message("Making Smart Connections..." + this.interval_count); - } - } - }, 10); - } - async render_note_connections(file) { - this.nearest = await this.plugin.find_note_connections(file); - } - clear_auto_searcher() { - if (this.search_timeout) { - clearTimeout(this.search_timeout); - this.search_timeout = null; - } - } - async search(search_text, results_only = false) { - const nearest = await this.plugin.api.search(search_text); - const nearest_context = `Selection: "${search_text.length > 100 ? search_text.substring(0, 100) + "..." : search_text}"`; - this.set_nearest(nearest, nearest_context, results_only); - } -}; -var SmartConnectionsViewApi = class { - constructor(app, plugin, view) { - this.app = app; - this.plugin = plugin; - this.view = view; - } - async search(search_text) { - return await this.plugin.api.search(search_text); - } - // trigger reload of embeddings file - async reload_embeddings_file() { - await this.plugin.init_vecs(); - await this.view.render_connections(); - } -}; -var ScSearchApi = class { - constructor(app, plugin) { - this.app = app; - this.plugin = plugin; - } - async search(search_text, filter = {}) { - filter = { - skip_sections: this.plugin.settings.skip_sections, - ...filter - }; - let nearest = []; - const resp = await this.plugin.request_embedding_from_input(search_text); - if (resp && resp.data && resp.data[0] && resp.data[0].embedding) { - nearest = this.plugin.smart_vec_lite.nearest(resp.data[0].embedding, filter); - } else { - new Obsidian.Notice("Smart Connections: Error getting embedding"); - } - return nearest; - } -}; -var SmartConnectionsSettingsTab = class extends Obsidian.PluginSettingTab { - constructor(app, plugin) { - super(app, plugin); - this.plugin = plugin; - } - display() { - const { - containerEl - } = this; - containerEl.empty(); - containerEl.createEl("h2", { - text: "Supporter Settings" - }); - containerEl.createEl("p", { - text: 'As a Smart Connections "Supporter", fast-track your PKM journey with priority perks and pioneering innovations.' - }); - const supporter_benefits_list = containerEl.createEl("ul"); - supporter_benefits_list.createEl("li", { - text: "Enjoy swift, top-priority support." - }); - supporter_benefits_list.createEl("li", { - text: "Gain early access to version 2 (includes local embedding model)." - }); - supporter_benefits_list.createEl("li", { - text: "Stay informed and engaged with exclusive supporter-only communications." - }); - new Obsidian.Setting(containerEl).setName("Supporter License Key").setDesc("Note: this is not required to use Smart Connections.").addText((text) => text.setPlaceholder("Enter your license_key").setValue(this.plugin.settings.license_key).onChange(async (value) => { - this.plugin.settings.license_key = value.trim(); - await this.plugin.saveSettings(true); - })); - new Obsidian.Setting(containerEl).setName("Get v2").setDesc("Get v2 (warning: very early beta release, likely to crash, please send issues directly to the supporter email for quick response)").addButton((button) => button.setButtonText("Get v2 (unstable)").onClick(async () => { - await this.plugin.update_to_v2(); - })); - new Obsidian.Setting(containerEl).setName("Sync Notes").setDesc("Make notes available via the Smart Connections ChatGPT Plugin. Respects exclusion settings configured below.").addButton((button) => button.setButtonText("Sync Notes").onClick(async () => { - await this.plugin.sync_notes(); - })); - new Obsidian.Setting(containerEl).setName("Become a Supporter").setDesc("Become a Supporter").addButton((button) => button.setButtonText("Become a Supporter").onClick(async () => { - const payment_pages = [ - "https://buy.stripe.com/9AQ5kO5QnbAWgGAbIY", - "https://buy.stripe.com/9AQ7sWemT48u1LGcN4" - ]; - if (!this.plugin.payment_page_index) { - this.plugin.payment_page_index = Math.round(Math.random()); - } - window.open(payment_pages[this.plugin.payment_page_index]); - })); - containerEl.createEl("h2", { - text: "OpenAI Settings" - }); - new Obsidian.Setting(containerEl).setName("OpenAI API Key").setDesc("Required: an OpenAI API key is currently required to use Smart Connections.").addText((text) => text.setPlaceholder("Enter your api_key").setValue(this.plugin.settings.api_key).onChange(async (value) => { - this.plugin.settings.api_key = value.trim(); - await this.plugin.saveSettings(true); - })); - new Obsidian.Setting(containerEl).setName("Test API Key").setDesc("Test API Key").addButton((button) => button.setButtonText("Test API Key").onClick(async () => { - const resp = await this.plugin.test_api_key(); - if (resp) { - new Obsidian.Notice("Smart Connections: API key is valid"); - } else { - new Obsidian.Notice("Smart Connections: API key is not working as expected!"); - } - })); - new Obsidian.Setting(containerEl).setName("Smart Chat Model").setDesc("Select a model to use with Smart Chat.").addDropdown((dropdown) => { - dropdown.addOption("gpt-3.5-turbo-16k", "gpt-3.5-turbo-16k"); - dropdown.addOption("gpt-4", "gpt-4 (limited access, 8k)"); - dropdown.addOption("gpt-3.5-turbo", "gpt-3.5-turbo (4k)"); - dropdown.addOption("gpt-4-1106-preview", "gpt-4-turbo (128k)"); - dropdown.onChange(async (value) => { - this.plugin.settings.smart_chat_model = value; - await this.plugin.saveSettings(); - }); - dropdown.setValue(this.plugin.settings.smart_chat_model); - }); - new Obsidian.Setting(containerEl).setName("Default Language").setDesc("Default language to use for Smart Chat. Changes which self-referential pronouns will trigger lookup of your notes.").addDropdown((dropdown) => { - const languages = Object.keys(SMART_TRANSLATION); - for (let i = 0; i < languages.length; i++) { - dropdown.addOption(languages[i], languages[i]); - } - dropdown.onChange(async (value) => { - this.plugin.settings.language = value; - await this.plugin.saveSettings(); - self_ref_pronouns_list.setText(this.get_self_ref_list()); - const chat_view = this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE).length > 0 ? this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE)[0].view : null; - if (chat_view) { - chat_view.new_chat(); - } - }); - dropdown.setValue(this.plugin.settings.language); - }); - const self_ref_pronouns_list = containerEl.createEl("span", { - text: this.get_self_ref_list() - }); - containerEl.createEl("h2", { - text: "Exclusions" - }); - new Obsidian.Setting(containerEl).setName("file_exclusions").setDesc("'Excluded file' matchers separated by a comma.").addText((text) => text.setPlaceholder("drawings,prompts/logs").setValue(this.plugin.settings.file_exclusions).onChange(async (value) => { - this.plugin.settings.file_exclusions = value; - await this.plugin.saveSettings(); - })); - new Obsidian.Setting(containerEl).setName("folder_exclusions").setDesc("'Excluded folder' matchers separated by a comma.").addText((text) => text.setPlaceholder("drawings,prompts/logs").setValue(this.plugin.settings.folder_exclusions).onChange(async (value) => { - this.plugin.settings.folder_exclusions = value; - await this.plugin.saveSettings(); - })); - new Obsidian.Setting(containerEl).setName("path_only").setDesc("'Path only' matchers separated by a comma.").addText((text) => text.setPlaceholder("drawings,prompts/logs").setValue(this.plugin.settings.path_only).onChange(async (value) => { - this.plugin.settings.path_only = value; - await this.plugin.saveSettings(); - })); - new Obsidian.Setting(containerEl).setName("header_exclusions").setDesc("'Excluded header' matchers separated by a comma. Works for 'blocks' only.").addText((text) => text.setPlaceholder("drawings,prompts/logs").setValue(this.plugin.settings.header_exclusions).onChange(async (value) => { - this.plugin.settings.header_exclusions = value; - await this.plugin.saveSettings(); - })); - containerEl.createEl("h2", { - text: "Display" - }); - new Obsidian.Setting(containerEl).setName("show_full_path").setDesc("Show full path in view.").addToggle((toggle) => toggle.setValue(this.plugin.settings.show_full_path).onChange(async (value) => { - this.plugin.settings.show_full_path = value; - await this.plugin.saveSettings(true); - })); - new Obsidian.Setting(containerEl).setName("expanded_view").setDesc("Expanded view by default.").addToggle((toggle) => toggle.setValue(this.plugin.settings.expanded_view).onChange(async (value) => { - this.plugin.settings.expanded_view = value; - await this.plugin.saveSettings(true); - })); - new Obsidian.Setting(containerEl).setName("group_nearest_by_file").setDesc("Group nearest by file.").addToggle((toggle) => toggle.setValue(this.plugin.settings.group_nearest_by_file).onChange(async (value) => { - this.plugin.settings.group_nearest_by_file = value; - await this.plugin.saveSettings(true); - })); - new Obsidian.Setting(containerEl).setName("view_open").setDesc("Open view on Obsidian startup.").addToggle((toggle) => toggle.setValue(this.plugin.settings.view_open).onChange(async (value) => { - this.plugin.settings.view_open = value; - await this.plugin.saveSettings(true); - })); - new Obsidian.Setting(containerEl).setName("chat_open").setDesc("Open view on Obsidian startup.").addToggle((toggle) => toggle.setValue(this.plugin.settings.chat_open).onChange(async (value) => { - this.plugin.settings.chat_open = value; - await this.plugin.saveSettings(true); - })); - containerEl.createEl("h2", { - text: "Advanced" - }); - new Obsidian.Setting(containerEl).setName("log_render").setDesc("Log render details to console (includes token_usage).").addToggle((toggle) => toggle.setValue(this.plugin.settings.log_render).onChange(async (value) => { - this.plugin.settings.log_render = value; - await this.plugin.saveSettings(true); - })); - new Obsidian.Setting(containerEl).setName("log_render_files").setDesc("Log embedded objects paths with log render (for debugging).").addToggle((toggle) => toggle.setValue(this.plugin.settings.log_render_files).onChange(async (value) => { - this.plugin.settings.log_render_files = value; - await this.plugin.saveSettings(true); - })); - new Obsidian.Setting(containerEl).setName("skip_sections").setDesc("Skips making connections to specific sections within notes. Warning: reduces usefulness for large files and requires 'Force Refresh' for sections to work in the future.").addToggle((toggle) => toggle.setValue(this.plugin.settings.skip_sections).onChange(async (value) => { - this.plugin.settings.skip_sections = value; - await this.plugin.saveSettings(true); - })); - containerEl.createEl("h3", { - text: "Test File Writing" - }); - containerEl.createEl("h3", { - text: "Manual Save" - }); - let manual_save_results = containerEl.createEl("div"); - new Obsidian.Setting(containerEl).setName("manual_save").setDesc("Save current embeddings").addButton((button) => button.setButtonText("Manual Save").onClick(async () => { - if (confirm("Are you sure you want to save your current embeddings?")) { - try { - await this.plugin.save_embeddings_to_file(true); - manual_save_results.innerHTML = "Embeddings saved successfully."; - } catch (e) { - manual_save_results.innerHTML = "Embeddings failed to save. Error: " + e; - } - } - })); - containerEl.createEl("h3", { - text: "Previously failed files" - }); - let failed_list = containerEl.createEl("div"); - this.draw_failed_files_list(failed_list); - containerEl.createEl("h3", { - text: "Force Refresh" - }); - new Obsidian.Setting(containerEl).setName("force_refresh").setDesc("WARNING: DO NOT use unless you know what you are doing! This will delete all of your current embeddings from OpenAI and trigger reprocessing of your entire vault!").addButton((button) => button.setButtonText("Force Refresh").onClick(async () => { - if (confirm("Are you sure you want to Force Refresh? By clicking yes you confirm that you understand the consequences of this action.")) { - await this.plugin.force_refresh_embeddings_file(); - } - })); - } - get_self_ref_list() { - return "Current: " + SMART_TRANSLATION[this.plugin.settings.language].pronous.join(", "); - } - draw_failed_files_list(failed_list) { - failed_list.empty(); - if (this.plugin.settings.failed_files.length > 0) { - failed_list.createEl("p", { - text: "The following files failed to process and will be skipped until manually retried." - }); - let list = failed_list.createEl("ul"); - for (let failed_file of this.plugin.settings.failed_files) { - list.createEl("li", { - text: failed_file - }); - } - new Obsidian.Setting(failed_list).setName("retry_failed_files").setDesc("Retry failed files only").addButton((button) => button.setButtonText("Retry failed files only").onClick(async () => { - failed_list.empty(); - failed_list.createEl("p", { - text: "Retrying failed files..." - }); - await this.plugin.retry_failed_files(); - this.draw_failed_files_list(failed_list); - })); - } else { - failed_list.createEl("p", { - text: "No failed files" - }); - } - } -}; -function line_is_heading(line) { - return line.indexOf("#") === 0 && ["#", " "].indexOf(line[1]) !== -1; -} -var SMART_CONNECTIONS_CHAT_VIEW_TYPE = "smart-connections-chat-view"; -var SmartConnectionsChatView = class extends Obsidian.ItemView { - constructor(leaf, plugin) { - super(leaf); - this.plugin = plugin; - this.active_elm = null; - this.active_stream = null; - this.brackets_ct = 0; - this.chat = null; - this.chat_box = null; - this.chat_container = null; - this.current_chat_ml = []; - this.files = []; - this.last_from = null; - this.message_container = null; - this.prevent_input = false; - } - getDisplayText() { - return "Smart Connections Chat"; - } - getIcon() { - return "message-square"; - } - getViewType() { - return SMART_CONNECTIONS_CHAT_VIEW_TYPE; - } - onOpen() { - this.new_chat(); - this.plugin.get_all_folders(); - } - onClose() { - this.chat.save_chat(); - this.app.workspace.unregisterHoverLinkSource(SMART_CONNECTIONS_CHAT_VIEW_TYPE); - } - render_chat() { - this.containerEl.empty(); - this.chat_container = this.containerEl.createDiv("sc-chat-container"); - this.render_top_bar(); - this.render_chat_box(); - this.render_chat_input(); - this.plugin.render_brand(this.containerEl, "chat"); - } - // render plus sign for clear button - render_top_bar() { - let top_bar_container = this.chat_container.createDiv("sc-top-bar-container"); - let chat_name = this.chat.name(); - let chat_name_input = top_bar_container.createEl("input", { - attr: { - type: "text", - value: chat_name - }, - cls: "sc-chat-name-input" - }); - chat_name_input.addEventListener("change", this.rename_chat.bind(this)); - let smart_view_btn = this.create_top_bar_button(top_bar_container, "Smart View", "smart-connections"); - smart_view_btn.addEventListener("click", this.open_smart_view.bind(this)); - let save_btn = this.create_top_bar_button(top_bar_container, "Save Chat", "save"); - save_btn.addEventListener("click", this.save_chat.bind(this)); - let history_btn = this.create_top_bar_button(top_bar_container, "Chat History", "history"); - history_btn.addEventListener("click", this.open_chat_history.bind(this)); - const new_chat_btn = this.create_top_bar_button(top_bar_container, "New Chat", "plus"); - new_chat_btn.addEventListener("click", this.new_chat.bind(this)); - } - async open_chat_history() { - const folder = await this.app.vault.adapter.list(".smart-connections/chats"); - this.files = folder.files.map((file) => { - return file.replace(".smart-connections/chats/", "").replace(".json", ""); - }); - if (!this.modal) - this.modal = new SmartConnectionsChatHistoryModal(this.app, this); - this.modal.open(); - } - create_top_bar_button(top_bar_container, title, icon = null) { - let btn = top_bar_container.createEl("button", { - attr: { - title - } - }); - if (icon) { - Obsidian.setIcon(btn, icon); - } else { - btn.innerHTML = title; - } - return btn; - } - // render new chat - new_chat() { - this.clear_chat(); - this.render_chat(); - this.new_messsage_bubble("assistant"); - this.active_elm.innerHTML = "

" + SMART_TRANSLATION[this.plugin.settings.language].initial_message + "

"; - } - // open a chat from the chat history modal - async open_chat(chat_id) { - this.clear_chat(); - await this.chat.load_chat(chat_id); - this.render_chat(); - for (let i = 0; i < this.chat.chat_ml.length; i++) { - await this.render_message(this.chat.chat_ml[i].content, this.chat.chat_ml[i].role); - } - } - // clear current chat state - clear_chat() { - if (this.chat) { - this.chat.save_chat(); - } - this.chat = new SmartConnectionsChatModel(this.plugin); - if (this.dotdotdot_interval) { - clearInterval(this.dotdotdot_interval); - } - this.current_chat_ml = []; - this.end_stream(); - } - rename_chat(event) { - let new_chat_name = event.target.value; - this.chat.rename_chat(new_chat_name); - } - // save current chat - save_chat() { - this.chat.save_chat(); - new Obsidian.Notice("[Smart Connections] Chat saved"); - } - open_smart_view() { - this.plugin.open_view(); - } - // render chat messages container - render_chat_box() { - this.chat_box = this.chat_container.createDiv("sc-chat-box"); - this.message_container = this.chat_box.createDiv("sc-message-container"); - } - // open file suggestion modal - open_file_suggestion_modal() { - if (!this.file_selector) - this.file_selector = new SmartConnectionsFileSelectModal(this.app, this); - this.file_selector.open(); - } - // open folder suggestion modal - async open_folder_suggestion_modal() { - if (!this.folder_selector) { - this.folder_selector = new SmartConnectionsFolderSelectModal(this.app, this); - } - this.folder_selector.open(); - } - // insert_selection from file suggestion modal - insert_selection(insert_text) { - let caret_pos = this.textarea.selectionStart; - let text_before = this.textarea.value.substring(0, caret_pos); - let text_after = this.textarea.value.substring(caret_pos, this.textarea.value.length); - this.textarea.value = text_before + insert_text + text_after; - this.textarea.selectionStart = caret_pos + insert_text.length; - this.textarea.selectionEnd = caret_pos + insert_text.length; - this.textarea.focus(); - } - // render chat textarea and button - render_chat_input() { - let chat_input = this.chat_container.createDiv("sc-chat-form"); - this.textarea = chat_input.createEl("textarea", { - cls: "sc-chat-input", - attr: { - placeholder: `Try "Based on my notes" or "Summarize [[this note]]" or "Important tasks in /folder/"` - } - }); - chat_input.addEventListener("keyup", (e) => { - if (["[", "/"].indexOf(e.key) === -1) - return; - const caret_pos = this.textarea.selectionStart; - if (e.key === "[") { - if (this.textarea.value[caret_pos - 2] === "[") { - this.open_file_suggestion_modal(); - return; - } - } else { - this.brackets_ct = 0; - } - if (e.key === "/") { - if (this.textarea.value.length === 1 || this.textarea.value[caret_pos - 2] === " ") { - this.open_folder_suggestion_modal(); - return; - } - } - }); - chat_input.addEventListener("keydown", (e) => { - if (e.key === "Enter" && e.shiftKey) { - e.preventDefault(); - if (this.prevent_input) { - console.log("wait until current response is finished"); - new Obsidian.Notice("[Smart Connections] Wait until current response is finished"); - return; - } - let user_input = this.textarea.value; - this.textarea.value = ""; - this.initialize_response(user_input); - } - this.textarea.style.height = "auto"; - this.textarea.style.height = this.textarea.scrollHeight + "px"; - }); - let button_container = chat_input.createDiv("sc-button-container"); - let abort_button = button_container.createEl("span", { attr: { id: "sc-abort-button", style: "display: none;" } }); - Obsidian.setIcon(abort_button, "square"); - abort_button.addEventListener("click", () => { - this.end_stream(); - }); - let button = button_container.createEl("button", { attr: { id: "sc-send-button" }, cls: "send-button" }); - button.innerHTML = "Send"; - button.addEventListener("click", () => { - if (this.prevent_input) { - console.log("wait until current response is finished"); - new Obsidian.Notice("Wait until current response is finished"); - return; - } - let user_input = this.textarea.value; - this.textarea.value = ""; - this.initialize_response(user_input); - }); - } - async initialize_response(user_input) { - this.set_streaming_ux(); - await this.render_message(user_input, "user"); - this.chat.new_message_in_thread({ - role: "user", - content: user_input - }); - await this.render_dotdotdot(); - if (this.chat.contains_internal_link(user_input)) { - this.chat.get_response_with_note_context(user_input, this); - return; - } - if (this.contains_self_referential_keywords(user_input) || this.chat.contains_folder_reference(user_input)) { - const context = await this.get_context_hyde(user_input); - const chatml = [ - { - role: "system", - // content: context_input - content: context - }, - { - role: "user", - content: user_input - } - ]; - this.request_chatgpt_completion({ messages: chatml, temperature: 0 }); - return; - } - this.request_chatgpt_completion(); - } - async render_dotdotdot() { - if (this.dotdotdot_interval) - clearInterval(this.dotdotdot_interval); - await this.render_message("...", "assistant"); - let dots = 0; - this.active_elm.innerHTML = "..."; - this.dotdotdot_interval = setInterval(() => { - dots++; - if (dots > 3) - dots = 1; - this.active_elm.innerHTML = ".".repeat(dots); - }, 500); - } - set_streaming_ux() { - this.prevent_input = true; - if (document.getElementById("sc-send-button")) - document.getElementById("sc-send-button").style.display = "none"; - if (document.getElementById("sc-abort-button")) - document.getElementById("sc-abort-button").style.display = "block"; - } - unset_streaming_ux() { - this.prevent_input = false; - if (document.getElementById("sc-send-button")) - document.getElementById("sc-send-button").style.display = ""; - if (document.getElementById("sc-abort-button")) - document.getElementById("sc-abort-button").style.display = "none"; - } - // check if includes keywords referring to one's own notes - contains_self_referential_keywords(user_input) { - const matches = user_input.match(this.plugin.self_ref_kw_regex); - if (matches) - return true; - return false; - } - // render message - async render_message(message, from = "assistant", append_last = false) { - if (this.dotdotdot_interval) { - clearInterval(this.dotdotdot_interval); - this.dotdotdot_interval = null; - this.active_elm.innerHTML = ""; - } - if (append_last) { - this.current_message_raw += message; - if (message.indexOf("\n") === -1) { - this.active_elm.innerHTML += message; - } else { - this.active_elm.innerHTML = ""; - await Obsidian.MarkdownRenderer.renderMarkdown(this.current_message_raw, this.active_elm, "?no-dataview", new Obsidian.Component()); - } - } else { - this.current_message_raw = ""; - if (this.chat.thread.length === 0 || this.last_from !== from) { - this.new_messsage_bubble(from); - } - this.active_elm.innerHTML = ""; - await Obsidian.MarkdownRenderer.renderMarkdown(message, this.active_elm, "?no-dataview", new Obsidian.Component()); - this.handle_links_in_message(); - this.render_message_action_buttons(message); - } - this.message_container.scrollTop = this.message_container.scrollHeight; - } - render_message_action_buttons(message) { - if (this.chat.context && this.chat.hyd) { - const context_view = this.active_elm.createEl("span", { - cls: "sc-msg-button", - attr: { - title: "Copy context to clipboard" - /* tooltip */ - } - }); - const this_hyd = this.chat.hyd; - Obsidian.setIcon(context_view, "eye"); - context_view.addEventListener("click", () => { - navigator.clipboard.writeText("```smart-connections\n" + this_hyd + "\n```\n"); - new Obsidian.Notice("[Smart Connections] Context code block copied to clipboard"); - }); - } - if (this.chat.context) { - const copy_prompt_button = this.active_elm.createEl("span", { - cls: "sc-msg-button", - attr: { - title: "Copy prompt to clipboard" - /* tooltip */ - } - }); - const this_context = this.chat.context.replace(/\`\`\`/g, " ```").trimLeft(); - Obsidian.setIcon(copy_prompt_button, "files"); - copy_prompt_button.addEventListener("click", () => { - navigator.clipboard.writeText("```prompt-context\n" + this_context + "\n```\n"); - new Obsidian.Notice("[Smart Connections] Context copied to clipboard"); - }); - } - const copy_button = this.active_elm.createEl("span", { - cls: "sc-msg-button", - attr: { - title: "Copy message to clipboard" - /* tooltip */ - } - }); - Obsidian.setIcon(copy_button, "copy"); - copy_button.addEventListener("click", () => { - navigator.clipboard.writeText(message.trimLeft()); - new Obsidian.Notice("[Smart Connections] Message copied to clipboard"); - }); - } - handle_links_in_message() { - const links = this.active_elm.querySelectorAll("a"); - if (links.length > 0) { - for (let i = 0; i < links.length; i++) { - const link = links[i]; - const link_text = link.getAttribute("data-href"); - link.addEventListener("mouseover", (event) => { - this.app.workspace.trigger("hover-link", { - event, - source: SMART_CONNECTIONS_CHAT_VIEW_TYPE, - hoverParent: link.parentElement, - targetEl: link, - // extract link text from a.data-href - linktext: link_text - }); - }); - link.addEventListener("click", (event) => { - const link_tfile = this.app.metadataCache.getFirstLinkpathDest(link_text, "/"); - const mod = Obsidian.Keymap.isModEvent(event); - let leaf = this.app.workspace.getLeaf(mod); - leaf.openFile(link_tfile); - }); - } - } - } - new_messsage_bubble(from) { - let message_el = this.message_container.createDiv(`sc-message ${from}`); - this.active_elm = message_el.createDiv("sc-message-content"); - this.last_from = from; - } - async request_chatgpt_completion(opts = {}) { - const chat_ml = opts.messages || opts.chat_ml || this.chat.prepare_chat_ml(); - console.log("chat_ml", chat_ml); - const max_total_tokens = Math.round(get_max_chars(this.plugin.settings.smart_chat_model) / 4); - console.log("max_total_tokens", max_total_tokens); - const curr_token_est = Math.round(JSON.stringify(chat_ml).length / 3); - console.log("curr_token_est", curr_token_est); - let max_available_tokens = max_total_tokens - curr_token_est; - if (max_available_tokens < 0) - max_available_tokens = 200; - else if (max_available_tokens > 4096) - max_available_tokens = 4096; - console.log("max_available_tokens", max_available_tokens); - opts = { - model: this.plugin.settings.smart_chat_model, - messages: chat_ml, - // max_tokens: 250, - max_tokens: max_available_tokens, - temperature: 0.3, - top_p: 1, - presence_penalty: 0, - frequency_penalty: 0, - stream: true, - stop: null, - n: 1, - // logit_bias: logit_bias, - ...opts - }; - if (opts.stream) { - const full_str = await new Promise((resolve, reject) => { - try { - const url = "https://api.openai.com/v1/chat/completions"; - this.active_stream = new ScStreamer(url, { - headers: { - "Content-Type": "application/json", - Authorization: `Bearer ${this.plugin.settings.api_key}` - }, - method: "POST", - payload: JSON.stringify(opts) - }); - let txt = ""; - this.active_stream.addEventListener("message", (e) => { - if (e.data != "[DONE]") { - const payload = JSON.parse(e.data); - const text = payload.choices[0].delta.content; - if (!text) { - return; - } - txt += text; - this.render_message(text, "assistant", true); - } else { - this.end_stream(); - resolve(txt); - } - }); - this.active_stream.addEventListener("readystatechange", (e) => { - if (e.readyState >= 2) { - console.log("ReadyState: " + e.readyState); - } - }); - this.active_stream.addEventListener("error", (e) => { - console.error(e); - new Obsidian.Notice("Smart Connections Error Streaming Response. See console for details."); - this.render_message("*API Error. See console logs for details.*", "assistant"); - this.end_stream(); - reject(e); - }); - this.active_stream.stream(); - } catch (err) { - console.error(err); - new Obsidian.Notice("Smart Connections Error Streaming Response. See console for details."); - this.end_stream(); - reject(err); - } - }); - await this.render_message(full_str, "assistant"); - this.chat.new_message_in_thread({ - role: "assistant", - content: full_str - }); - return; - } else { - try { - const response = await (0, Obsidian.requestUrl)({ - url: `https://api.openai.com/v1/chat/completions`, - method: "POST", - headers: { - Authorization: `Bearer ${this.plugin.settings.api_key}`, - "Content-Type": "application/json" - }, - contentType: "application/json", - body: JSON.stringify(opts), - throw: false - }); - return JSON.parse(response.text).choices[0].message.content; - } catch (err) { - new Obsidian.Notice(`Smart Connections API Error :: ${err}`); - } - } - } - end_stream() { - if (this.active_stream) { - this.active_stream.close(); - this.active_stream = null; - } - this.unset_streaming_ux(); - if (this.dotdotdot_interval) { - clearInterval(this.dotdotdot_interval); - this.dotdotdot_interval = null; - this.active_elm.parentElement.remove(); - this.active_elm = null; - } - } - async get_context_hyde(user_input) { - this.chat.reset_context(); - const hyd_input = `Anticipate what the user is seeking. Respond in the form of a hypothetical note written by the user. The note may contain statements as paragraphs, lists, or checklists in markdown format with no headings. Please respond with one hypothetical note and abstain from any other commentary. Use the format: PARENT FOLDER NAME > CHILD FOLDER NAME > FILE NAME > HEADING 1 > HEADING 2 > HEADING 3: HYPOTHETICAL NOTE CONTENTS.`; - const chatml = [ - { - role: "system", - content: hyd_input - }, - { - role: "user", - content: user_input - } - ]; - const hyd = await this.request_chatgpt_completion({ - messages: chatml, - stream: false, - temperature: 0, - max_tokens: 137 - }); - this.chat.hyd = hyd; - let filter = {}; - if (this.chat.contains_folder_reference(user_input)) { - const folder_refs = this.chat.get_folder_references(user_input); - if (folder_refs) { - filter = { - path_begins_with: folder_refs - }; - } - } - let nearest = await this.plugin.api.search(hyd, filter); - console.log("nearest", nearest.length); - nearest = this.get_nearest_until_next_dev_exceeds_std_dev(nearest); - console.log("nearest after std dev slice", nearest.length); - nearest = this.sort_by_len_adjusted_similarity(nearest); - return await this.get_context_for_prompt(nearest); - } - sort_by_len_adjusted_similarity(nearest) { - nearest = nearest.sort((a, b) => { - const a_score = a.similarity / a.len; - const b_score = b.similarity / b.len; - if (a_score > b_score) - return -1; - if (a_score < b_score) - return 1; - return 0; - }); - return nearest; - } - get_nearest_until_next_dev_exceeds_std_dev(nearest) { - const sim = nearest.map((n) => n.similarity); - const mean = sim.reduce((a, b) => a + b) / sim.length; - let std_dev = Math.sqrt(sim.map((x) => Math.pow(x - mean, 2)).reduce((a, b) => a + b) / sim.length); - let slice_i = 0; - while (slice_i < nearest.length) { - const next = nearest[slice_i + 1]; - if (next) { - const next_dev = Math.abs(next.similarity - nearest[slice_i].similarity); - if (next_dev > std_dev) { - if (slice_i < 3) - std_dev = std_dev * 1.5; - else - break; - } - } - slice_i++; - } - nearest = nearest.slice(0, slice_i + 1); - return nearest; - } - // this.test_get_nearest_until_next_dev_exceeds_std_dev(); - // // test get_nearest_until_next_dev_exceeds_std_dev - // test_get_nearest_until_next_dev_exceeds_std_dev() { - // const nearest = [{similarity: 0.99}, {similarity: 0.98}, {similarity: 0.97}, {similarity: 0.96}, {similarity: 0.95}, {similarity: 0.94}, {similarity: 0.93}, {similarity: 0.92}, {similarity: 0.91}, {similarity: 0.9}, {similarity: 0.79}, {similarity: 0.78}, {similarity: 0.77}, {similarity: 0.76}, {similarity: 0.75}, {similarity: 0.74}, {similarity: 0.73}, {similarity: 0.72}]; - // const result = this.get_nearest_until_next_dev_exceeds_std_dev(nearest); - // if(result.length !== 10){ - // console.error("get_nearest_until_next_dev_exceeds_std_dev failed", result); - // } - // } - async get_context_for_prompt(nearest) { - let context = []; - const MAX_SOURCES = this.plugin.settings.smart_chat_model === "gpt-4-1106-preview" ? 42 : 20; - const MAX_CHARS = get_max_chars(this.plugin.settings.smart_chat_model) / 2; - let char_accum = 0; - for (let i = 0; i < nearest.length; i++) { - if (context.length >= MAX_SOURCES) - break; - if (char_accum >= MAX_CHARS) - break; - if (typeof nearest[i].link !== "string") - continue; - const breadcrumbs = nearest[i].link.replace(/#/g, " > ").replace(".md", "").replace(/\//g, " > "); - let new_context = `${breadcrumbs}: -`; - const max_available_chars = MAX_CHARS - char_accum - new_context.length; - if (nearest[i].link.indexOf("#") !== -1) { - new_context += await this.plugin.block_retriever(nearest[i].link, { max_chars: max_available_chars }); - } else { - new_context += await this.plugin.file_retriever(nearest[i].link, { max_chars: max_available_chars }); - } - char_accum += new_context.length; - context.push({ - link: nearest[i].link, - text: new_context - }); - } - console.log("context sources: " + context.length); - console.log("total context tokens: ~" + Math.round(char_accum / 3.5)); - this.chat.context = `Anticipate the type of answer desired by the user. Imagine the following ${context.length} notes were written by the user and contain all the necessary information to answer the user's question. Begin responses with "${SMART_TRANSLATION[this.plugin.settings.language].prompt}..."`; - for (let i = 0; i < context.length; i++) { - this.chat.context += ` ----BEGIN #${i + 1}--- -${context[i].text} ----END #${i + 1}---`; - } - return this.chat.context; - } -}; -function get_max_chars(model = "gpt-3.5-turbo") { - const MAX_CHAR_MAP = { - "gpt-3.5-turbo-16k": 48e3, - "gpt-4": 24e3, - "gpt-3.5-turbo": 12e3, - "gpt-4-1106-preview": 2e5 - }; - return MAX_CHAR_MAP[model]; -} -var SmartConnectionsChatModel = class { - constructor(plugin) { - this.app = plugin.app; - this.plugin = plugin; - this.chat_id = null; - this.chat_ml = []; - this.context = null; - this.hyd = null; - this.thread = []; - } - async save_chat() { - if (this.thread.length === 0) - return; - if (!await this.app.vault.adapter.exists(".smart-connections/chats")) { - await this.app.vault.adapter.mkdir(".smart-connections/chats"); - } - if (!this.chat_id) { - this.chat_id = this.name() + "\u2014" + this.get_file_date_string(); - } - if (!this.chat_id.match(/^[a-zA-Z0-9_—\- ]+$/)) { - console.log("Invalid chat_id: " + this.chat_id); - new Obsidian.Notice("[Smart Connections] Failed to save chat. Invalid chat_id: '" + this.chat_id + "'"); - } - const chat_file = this.chat_id + ".json"; - this.app.vault.adapter.write( - ".smart-connections/chats/" + chat_file, - JSON.stringify(this.thread, null, 2) - ); - } - async load_chat(chat_id) { - this.chat_id = chat_id; - const chat_file = this.chat_id + ".json"; - let chat_json = await this.app.vault.adapter.read( - ".smart-connections/chats/" + chat_file - ); - this.thread = JSON.parse(chat_json); - this.chat_ml = this.prepare_chat_ml(); - } - // prepare chat_ml from chat - // gets the last message of each turn unless turn_variation_offsets=[[turn_index,variation_index]] is specified in offset - prepare_chat_ml(turn_variation_offsets = []) { - if (turn_variation_offsets.length === 0) { - this.chat_ml = this.thread.map((turn) => { - return turn[turn.length - 1]; - }); - } else { - let turn_variation_index = []; - for (let i = 0; i < turn_variation_offsets.length; i++) { - turn_variation_index[turn_variation_offsets[i][0]] = turn_variation_offsets[i][1]; - } - this.chat_ml = this.thread.map((turn, turn_index) => { - if (turn_variation_index[turn_index] !== void 0) { - return turn[turn_variation_index[turn_index]]; - } - return turn[turn.length - 1]; - }); - } - this.chat_ml = this.chat_ml.map((message) => { - return { - role: message.role, - content: message.content - }; - }); - return this.chat_ml; - } - last() { - return this.thread[this.thread.length - 1][this.thread[this.thread.length - 1].length - 1]; - } - last_from() { - return this.last().role; - } - // returns user_input or completion - last_message() { - return this.last().content; - } - // message={} - // add new message to thread - new_message_in_thread(message, turn = -1) { - if (this.context) { - message.context = this.context; - this.context = null; - } - if (this.hyd) { - message.hyd = this.hyd; - this.hyd = null; - } - if (turn === -1) { - this.thread.push([message]); - } else { - this.thread[turn].push(message); - } - } - reset_context() { - this.context = null; - this.hyd = null; - } - async rename_chat(new_name) { - if (this.chat_id && await this.app.vault.adapter.exists(".smart-connections/chats/" + this.chat_id + ".json")) { - new_name = this.chat_id.replace(this.name(), new_name); - await this.app.vault.adapter.rename( - ".smart-connections/chats/" + this.chat_id + ".json", - ".smart-connections/chats/" + new_name + ".json" - ); - this.chat_id = new_name; - } else { - this.chat_id = new_name + "\u2014" + this.get_file_date_string(); - await this.save_chat(); - } - } - name() { - if (this.chat_id) { - return this.chat_id.replace(/—[^—]*$/, ""); - } - return "UNTITLED"; - } - get_file_date_string() { - return (/* @__PURE__ */ new Date()).toISOString().replace(/(T|:|\..*)/g, " ").trim(); - } - // get response from with note context - async get_response_with_note_context(user_input, chat_view) { - let system_input = "Imagine the following notes were written by the user and contain the necessary information to synthesize a useful answer the user's query:\n"; - const notes = this.extract_internal_links(user_input); - let max_chars = get_max_chars(this.plugin.settings.smart_chat_model); - for (let i = 0; i < notes.length; i++) { - const this_max_chars = notes.length - i > 1 ? Math.floor(max_chars / (notes.length - i)) : max_chars; - const note_content = await this.get_note_contents(notes[i], { char_limit: this_max_chars }); - system_input += `---BEGIN NOTE: [[${notes[i].basename}]]--- -`; - system_input += note_content; - system_input += `---END NOTE--- -`; - max_chars -= note_content.length; - if (max_chars <= 0) - break; - } - this.context = system_input; - const chatml = [ - { - role: "system", - content: system_input - }, - { - role: "user", - content: user_input - } - ]; - chat_view.request_chatgpt_completion({ messages: chatml, temperature: 0 }); - } - // check if contains internal link - contains_internal_link(user_input) { - if (user_input.indexOf("[[") === -1) - return false; - if (user_input.indexOf("]]") === -1) - return false; - return true; - } - // check if contains folder reference (ex. /folder/, or /folder/subfolder/) - contains_folder_reference(user_input) { - if (user_input.indexOf("/") === -1) - return false; - if (user_input.indexOf("/") === user_input.lastIndexOf("/")) - return false; - return true; - } - // get folder references from user input - get_folder_references(user_input) { - const folders = this.plugin.folders.slice(); - const matches = folders.sort((a, b) => b.length - a.length).map((folder) => { - if (user_input.indexOf(folder) !== -1) { - user_input = user_input.replace(folder, ""); - return folder; - } - return false; - }).filter((folder) => folder); - console.log(matches); - if (matches) - return matches; - return false; - } - // extract internal links - extract_internal_links(user_input) { - const matches = user_input.match(/\[\[(.*?)\]\]/g); - console.log(matches); - if (matches) - return matches.map((match) => { - return this.app.metadataCache.getFirstLinkpathDest(match.replace("[[", "").replace("]]", ""), "/"); - }); - return []; - } - // get context from internal links - async get_note_contents(note, opts = {}) { - opts = { - char_limit: 1e4, - ...opts - }; - if (!(note instanceof Obsidian.TFile)) - return ""; - let file_content = await this.app.vault.cachedRead(note); - if (file_content.indexOf("```dataview") > -1) { - file_content = await this.render_dataview_queries(file_content, note.path, opts); - } - file_content = file_content.substring(0, opts.char_limit); - return file_content; - } - async render_dataview_queries(file_content, note_path, opts = {}) { - opts = { - char_limit: null, - ...opts - }; - const dataview_api = window["DataviewAPI"]; - if (!dataview_api) - return file_content; - const dataview_code_blocks = file_content.match(/```dataview(.*?)```/gs); - for (let i = 0; i < dataview_code_blocks.length; i++) { - if (opts.char_limit && opts.char_limit < file_content.indexOf(dataview_code_blocks[i])) - break; - const dataview_code_block = dataview_code_blocks[i]; - const dataview_code_block_content = dataview_code_block.replace("```dataview", "").replace("```", ""); - const dataview_query_result = await dataview_api.queryMarkdown(dataview_code_block_content, note_path, null); - if (dataview_query_result.successful) { - file_content = file_content.replace(dataview_code_block, dataview_query_result.value); - } - } - return file_content; - } -}; -var SmartConnectionsChatHistoryModal = class extends Obsidian.FuzzySuggestModal { - constructor(app, view, files) { - super(app); - this.app = app; - this.view = view; - this.setPlaceholder("Type the name of a chat session..."); - } - getItems() { - if (!this.view.files) { - return []; - } - return this.view.files; - } - getItemText(item) { - if (item.indexOf("UNTITLED") === -1) { - item.replace(/—[^—]*$/, ""); - } - return item; - } - onChooseItem(session) { - this.view.open_chat(session); - } -}; -var SmartConnectionsFileSelectModal = class extends Obsidian.FuzzySuggestModal { - constructor(app, view) { - super(app); - this.app = app; - this.view = view; - this.setPlaceholder("Type the name of a file..."); - } - getItems() { - return this.app.vault.getMarkdownFiles().sort((a, b) => a.basename.localeCompare(b.basename)); - } - getItemText(item) { - return item.basename; - } - onChooseItem(file) { - this.view.insert_selection(file.basename + "]] "); - } -}; -var SmartConnectionsFolderSelectModal = class extends Obsidian.FuzzySuggestModal { - constructor(app, view) { - super(app); - this.app = app; - this.view = view; - this.setPlaceholder("Type the name of a folder..."); - } - getItems() { - return this.view.plugin.folders; - } - getItemText(item) { - return item; - } - onChooseItem(folder) { - this.view.insert_selection(folder + "/ "); - } -}; -var ScStreamer = class { - // constructor - constructor(url, options) { - options = options || {}; - this.url = url; - this.method = options.method || "GET"; - this.headers = options.headers || {}; - this.payload = options.payload || null; - this.withCredentials = options.withCredentials || false; - this.listeners = {}; - this.readyState = this.CONNECTING; - this.progress = 0; - this.chunk = ""; - this.xhr = null; - this.FIELD_SEPARATOR = ":"; - this.INITIALIZING = -1; - this.CONNECTING = 0; - this.OPEN = 1; - this.CLOSED = 2; - } - // addEventListener - addEventListener(type, listener) { - if (!this.listeners[type]) { - this.listeners[type] = []; - } - if (this.listeners[type].indexOf(listener) === -1) { - this.listeners[type].push(listener); - } - } - // removeEventListener - removeEventListener(type, listener) { - if (!this.listeners[type]) { - return; - } - let filtered = []; - for (let i = 0; i < this.listeners[type].length; i++) { - if (this.listeners[type][i] !== listener) { - filtered.push(this.listeners[type][i]); - } - } - if (this.listeners[type].length === 0) { - delete this.listeners[type]; - } else { - this.listeners[type] = filtered; - } - } - // dispatchEvent - dispatchEvent(event) { - if (!event) { - return true; - } - event.source = this; - let onHandler = "on" + event.type; - if (this.hasOwnProperty(onHandler)) { - this[onHandler].call(this, event); - if (event.defaultPrevented) { - return false; - } - } - if (this.listeners[event.type]) { - return this.listeners[event.type].every(function(callback) { - callback(event); - return !event.defaultPrevented; - }); - } - return true; - } - // _setReadyState - _setReadyState(state) { - let event = new CustomEvent("readyStateChange"); - event.readyState = state; - this.readyState = state; - this.dispatchEvent(event); - } - // _onStreamFailure - _onStreamFailure(e) { - let event = new CustomEvent("error"); - event.data = e.currentTarget.response; - this.dispatchEvent(event); - this.close(); - } - // _onStreamAbort - _onStreamAbort(e) { - let event = new CustomEvent("abort"); - this.close(); - } - // _onStreamProgress - _onStreamProgress(e) { - if (!this.xhr) { - return; - } - if (this.xhr.status !== 200) { - this._onStreamFailure(e); - return; - } - if (this.readyState === this.CONNECTING) { - this.dispatchEvent(new CustomEvent("open")); - this._setReadyState(this.OPEN); - } - let data = this.xhr.responseText.substring(this.progress); - this.progress += data.length; - data.split(/(\r\n|\r|\n){2}/g).forEach(function(part) { - if (part.trim().length === 0) { - this.dispatchEvent(this._parseEventChunk(this.chunk.trim())); - this.chunk = ""; - } else { - this.chunk += part; - } - }.bind(this)); - } - // _onStreamLoaded - _onStreamLoaded(e) { - this._onStreamProgress(e); - this.dispatchEvent(this._parseEventChunk(this.chunk)); - this.chunk = ""; - } - // _parseEventChunk - _parseEventChunk(chunk) { - if (!chunk || chunk.length === 0) { - return null; - } - let e = { id: null, retry: null, data: "", event: "message" }; - chunk.split(/(\r\n|\r|\n)/).forEach(function(line) { - line = line.trimRight(); - let index = line.indexOf(this.FIELD_SEPARATOR); - if (index <= 0) { - return; - } - let field = line.substring(0, index); - if (!(field in e)) { - return; - } - let value = line.substring(index + 1).trimLeft(); - if (field === "data") { - e[field] += value; - } else { - e[field] = value; - } - }.bind(this)); - let event = new CustomEvent(e.event); - event.data = e.data; - event.id = e.id; - return event; - } - // _checkStreamClosed - _checkStreamClosed() { - if (!this.xhr) { - return; - } - if (this.xhr.readyState === XMLHttpRequest.DONE) { - this._setReadyState(this.CLOSED); - } - } - // stream - stream() { - this._setReadyState(this.CONNECTING); - this.xhr = new XMLHttpRequest(); - this.xhr.addEventListener("progress", this._onStreamProgress.bind(this)); - this.xhr.addEventListener("load", this._onStreamLoaded.bind(this)); - this.xhr.addEventListener("readystatechange", this._checkStreamClosed.bind(this)); - this.xhr.addEventListener("error", this._onStreamFailure.bind(this)); - this.xhr.addEventListener("abort", this._onStreamAbort.bind(this)); - this.xhr.open(this.method, this.url); - for (let header in this.headers) { - this.xhr.setRequestHeader(header, this.headers[header]); - } - this.xhr.withCredentials = this.withCredentials; - this.xhr.send(this.payload); - } - // close - close() { - if (this.readyState === this.CLOSED) { - return; - } - this.xhr.abort(); - this.xhr = null; - this._setReadyState(this.CLOSED); - } -}; -module.exports = SmartConnectionsPlugin; -//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../vec_lite/vec_lite.js", "../src/index.js"],
  "sourcesContent": ["class VecLite {\n  constructor(config) {\n    // set default config\n    this.config = {\n      file_name: \"embeddings-3.json\",\n      folder_path: \".vec_lite\",\n      exists_adapter: null,\n      mkdir_adapter: null,\n      read_adapter: null,\n      rename_adapter: null,\n      stat_adapter: null,\n      write_adapter: null,\n      ...config\n    };\n    this.file_name = this.config.file_name;\n    this.folder_path = config.folder_path;\n    this.file_path = this.folder_path + \"/\" + this.file_name;\n    // get folder path\n    this.embeddings = false;\n  }\n  async file_exists(path) {\n    if (this.config.exists_adapter) {\n      return await this.config.exists_adapter(path);\n    } else {\n      // todo handle with fs\n      throw new Error(\"exists_adapter not set\");\n    }\n  }\n  async mkdir(path) {\n    if (this.config.mkdir_adapter) {\n      return await this.config.mkdir_adapter(path);\n    } else {\n      // todo handle with fs\n      throw new Error(\"mkdir_adapter not set\");\n    }\n  }\n  async read_file(path) {\n    if (this.config.read_adapter) {\n      return await this.config.read_adapter(path);\n    } else {\n      // todo handle with fs\n      throw new Error(\"read_adapter not set\");\n    }\n  }\n  async rename(old_path, new_path) {\n    if (this.config.rename_adapter) {\n      return await this.config.rename_adapter(old_path, new_path);\n    } else {\n      // todo handle with fs\n      throw new Error(\"rename_adapter not set\");\n    }\n  }\n  async stat(path) {\n    if (this.config.stat_adapter) {\n      return await this.config.stat_adapter(path);\n    } else {\n      // todo handle with fs\n      throw new Error(\"stat_adapter not set\");\n    }\n  }\n  async write_file(path, data) {\n    if (this.config.write_adapter) {\n      return await this.config.write_adapter(path, data);\n    } else {\n      // todo handle with fs\n      throw new Error(\"write_adapter not set\");\n    }\n  }\n  async load(retries = 0) {\n    try {\n      const embeddings_file = await this.read_file(this.file_path);\n      // loaded embeddings from file\n      this.embeddings = JSON.parse(embeddings_file);\n      console.log(\"loaded embeddings file: \"+this.file_path);\n      return true;\n    } catch (error) {\n      // retry if error up to 3 times\n      if (retries < 3) {\n        console.log(\"retrying load()\");\n        // increase wait time between retries\n        await new Promise(r => setTimeout(r, 1000 + (1000 * retries)));\n        return await this.load(retries + 1);\n      // } else if (retries === 3) {\n      //   // check for embeddings-2.json file\n      //   const embeddings_2_file_path = this.folder_path + \"/embeddings-2.json\";\n      //   const embeddings_2_file_exists = await this.file_exists(embeddings_2_file_path);\n      //   if (embeddings_2_file_exists) {\n      //     await this.migrate_embeddings_v2_to_v3();\n      //     return await this.load(retries + 1);\n      //   }\n      }\n      console.log(\"failed to load embeddings file, prompt user to initiate bulk embed\");\n      return false;\n    }\n  }\n\n  async init_embeddings_file() {\n    // check if folder exists\n    if (!(await this.file_exists(this.folder_path))) {\n      // create folder\n      await this.mkdir(this.folder_path);\n      console.log(\"created folder: \"+this.folder_path);\n    } else {\n      console.log(\"folder already exists: \"+this.folder_path);\n    }\n    // check if embeddings file exists\n    if (!(await this.file_exists(this.file_path))) {\n      // create embeddings file\n      await this.write_file(this.file_path, \"{}\");\n      console.log(\"created embeddings file: \"+this.file_path);\n    } else {\n      console.log(\"embeddings file already exists: \"+this.file_path);\n    }\n  }\n\n  async save() {\n    const embeddings = JSON.stringify(this.embeddings);\n    // check if embeddings file exists\n    const embeddings_file_exists = await this.file_exists(this.file_path);\n    // if embeddings file exists then check if new embeddings file size is significantly smaller than existing embeddings file size\n    if (embeddings_file_exists) {\n      // esitmate file size of embeddings\n      const new_file_size = embeddings.length;\n      // get existing file size\n      const existing_file_size = await this.stat(this.file_path).then((stat) => stat.size);\n      // console.log(\"new file size: \"+new_file_size);\n      // console.log(\"existing file size: \"+existing_file_size);\n      // if new file size is at least 50% of existing file size then write embeddings to file\n      if (new_file_size > (existing_file_size * 0.5)) {\n        // write embeddings to file\n        await this.write_file(this.file_path, embeddings);\n        console.log(\"embeddings file size: \" + new_file_size + \" bytes\");\n      } else {\n        // if new file size is significantly smaller than existing file size then throw error\n        // show warning message including file sizes\n        const warning_message = [\n          \"Warning: New embeddings file size is significantly smaller than existing embeddings file size.\",\n          \"Aborting to prevent possible loss of embeddings data.\",\n          \"New file size: \" + new_file_size + \" bytes.\",\n          \"Existing file size: \" + existing_file_size + \" bytes.\",\n          \"Restarting Obsidian may fix this.\"\n        ];\n        console.log(warning_message.join(\" \"));\n        // save embeddings to file named unsaved-embeddings.json\n        await this.write_file(this.folder_path+\"/unsaved-embeddings.json\", embeddings);\n        throw new Error(\"Error: New embeddings file size is significantly smaller than existing embeddings file size. Aborting to prevent possible loss of embeddings data.\");\n      }\n    } else {\n      await this.init_embeddings_file();\n      return await this.save();\n    }\n    return true;\n  }\n  cos_sim(vector1, vector2) {\n    let dotProduct = 0;\n    let normA = 0;\n    let normB = 0;\n    for (let i = 0; i < vector1.length; i++) {\n      dotProduct += vector1[i] * vector2[i];\n      normA += vector1[i] * vector1[i];\n      normB += vector2[i] * vector2[i];\n    }\n    if (normA === 0 || normB === 0) {\n      return 0;\n    } else {\n      return dotProduct / (Math.sqrt(normA) * Math.sqrt(normB));\n    }\n  }\n  nearest(to_vec, filter = {}) {\n    filter = {\n      results_count: 30,\n      ...filter\n    };\n    let nearest = [];\n    const from_keys = Object.keys(this.embeddings);\n    // this.render_log.total_embeddings = from_keys.length;\n    for (let i = 0; i < from_keys.length; i++) {\n      // if this.settings.skip_sections is true\n      if (filter.skip_sections) {\n        const from_path = this.embeddings[from_keys[i]].meta.path;\n        if (from_path.indexOf(\"#\") > -1) continue; // skip if contains # indicating block (section)\n\n        // TODO: consider using presence of meta.parent to skip files (faster checking?)\n      }\n      if (filter.skip_key) {\n        if (filter.skip_key === from_keys[i]) continue; // skip matching to current note\n        if (filter.skip_key === this.embeddings[from_keys[i]].meta.parent) continue; // skip if filter.skip_key matches meta.parent\n      }\n      // if filter.path_begins_with is set (folder filter)\n      if (filter.path_begins_with) {\n        // if type is string & meta.path does not begin with filter.path_begins_with, skip\n        if (typeof filter.path_begins_with === \"string\" && !this.embeddings[from_keys[i]].meta.path.startsWith(filter.path_begins_with)) continue;\n        // if type is array & meta.path does not begin with any of the filter.path_begins_with, skip\n        if (Array.isArray(filter.path_begins_with) && !filter.path_begins_with.some((path) => this.embeddings[from_keys[i]].meta.path.startsWith(path))) continue;\n      }\n\n      nearest.push({\n        link: this.embeddings[from_keys[i]].meta.path,\n        similarity: this.cos_sim(to_vec, this.embeddings[from_keys[i]].vec),\n        size: this.embeddings[from_keys[i]].meta.size,\n      });\n    }\n    // sort array by cosine similarity\n    nearest.sort(function (a, b) {\n      return b.similarity - a.similarity;\n    });\n    // console.log(nearest);\n    // limit to N nearest connections\n    nearest = nearest.slice(0, filter.results_count);\n    return nearest;\n  }\n  find_nearest_embeddings(to_vec, filter={}) {\n    const default_filter = {\n      max: this.max_sources,\n    };\n    filter = {...default_filter, ...filter};\n    // handle if to_vec is an array of vectors\n    // let nearest = [];\n    if(Array.isArray(to_vec) && to_vec.length !== this.vec_len){\n      this.nearest = {};\n      for(let i = 0; i < to_vec.length; i++){\n        // nearest = nearest.concat(this.find_nearest_embeddings(to_vec[i], {\n        //   max: Math.floor(filter.max / to_vec.length)\n        // }));\n        this.find_nearest_embeddings(to_vec[i], {\n          max: Math.floor(filter.max / to_vec.length)\n        });\n      }\n    }else{\n      const from_keys = Object.keys(this.embeddings);\n      for (let i = 0; i < from_keys.length; i++) {\n        if(this.validate_type(this.embeddings[from_keys[i]])) continue;\n        const sim = this.computeCosineSimilarity(to_vec, this.embeddings[from_keys[i]].vec);\n        if(this.nearest[from_keys[i]]){ // if already computed, use cached value\n          this.nearest[from_keys[i]] += sim;\n        }else{\n          this.nearest[from_keys[i]] = sim;\n        }\n      }\n    }\n    // initiate nearest array\n    let nearest = Object.keys(this.nearest).map(key => {\n      return {\n        key: key,\n        similarity: this.nearest[key],\n      }\n    });\n    // sort array by cosine similarity\n    nearest = this.sort_by_similarity(nearest);\n    nearest = nearest.slice(0, filter.max);\n    // add link and length to remaining nearest\n    nearest = nearest.map(item => {\n      return {\n        link: this.embeddings[item.key].meta.path,\n        similarity: item.similarity,\n        len: this.embeddings[item.key].meta.len || this.embeddings[item.key].meta.size,\n      }\n    });\n    return nearest;\n  }\n  sort_by_similarity(nearest) {\n    return nearest.sort(function (a, b) {\n      const a_score = a.similarity;\n      const b_score = b.similarity;\n      // if a is greater than b, return -1\n      if (a_score > b_score)\n        return -1;\n      // if a is less than b, return 1\n      if (a_score < b_score)\n        return 1;\n      // if a is equal to b, return 0\n      return 0;\n    });\n  }\n  // check if key from embeddings exists in files\n  clean_up_embeddings(files) {\n    console.log(\"cleaning up embeddings\");\n    const keys = Object.keys(this.embeddings);\n    let deleted_embeddings = 0;\n    for (const key of keys) {\n      // console.log(\"key: \"+key);\n      const path = this.embeddings[key].meta.path;\n      // if no key starts with file path\n      if(!files.find(file => path.startsWith(file.path))) {\n        // delete key if it doesn't exist\n        delete this.embeddings[key];\n        deleted_embeddings++;\n        // console.log(\"deleting (deleted file): \" + key);\n        continue;\n      }\n      // if key contains '#'\n      if(path.indexOf(\"#\") > -1) {\n        const parent_key = this.embeddings[key].meta.parent;\n        // if parent_key missing from embeddings then delete key\n        if(!this.embeddings[parent_key]){\n          // delete key\n          delete this.embeddings[key];\n          deleted_embeddings++;\n          // console.log(\"deleting (missing parent)\");\n          continue;\n        }\n        // if parent_key missing meta then delete key\n        if(!this.embeddings[parent_key].meta){\n          // delete key\n          delete this.embeddings[key];\n          deleted_embeddings++;\n          // console.log(\"deleting (parent missing meta)\");\n          continue;\n        }\n        // if parent_key missing children then delete key\n        // if parent_key children doesn't include key then delete key\n        if(this.embeddings[parent_key].meta.children && (this.embeddings[parent_key].meta.children.indexOf(key) < 0)) {\n          // delete key\n          delete this.embeddings[key];\n          deleted_embeddings++;\n          // console.log(\"deleting (not present in parent's children)\");\n          continue;\n        }\n      }\n    }\n    return {deleted_embeddings: deleted_embeddings, total_embeddings: keys.length};\n  }\n\n  get(key) {\n    return this.embeddings[key] || null;\n  }\n  get_meta(key) {\n    const embedding = this.get(key);\n    if(embedding && embedding.meta) {\n      return embedding.meta;\n    }\n    return null;\n  }\n  get_mtime(key) {\n    const meta = this.get_meta(key);\n    if(meta && meta.mtime) {\n      return meta.mtime;\n    }\n    return null;\n  }\n  get_hash(key) {\n    const meta = this.get_meta(key);\n    if(meta && meta.hash) {\n      return meta.hash;\n    }\n    return null;\n  }\n  get_size(key) {\n    const meta = this.get_meta(key);\n    if(meta && meta.size) {\n      return meta.size;\n    }\n    return null;\n  }\n  get_children(key) {\n    const meta = this.get_meta(key);\n    if(meta && meta.children) {\n      return meta.children;\n    }\n    return null;\n  }\n  get_vec(key) {\n    const embedding = this.get(key);\n    if(embedding && embedding.vec) {\n      return embedding.vec;\n    }\n    return null;\n  }\n  save_embedding(key, vec, meta) {\n    this.embeddings[key] = {\n      vec: vec,\n      meta: meta,\n    };\n  }\n  mtime_is_current(key, source_mtime) {\n    const mtime = this.get_mtime(key);\n    if(mtime && mtime >= source_mtime) {\n      return true;\n    }\n    return false;\n  }\n\n  async force_refresh() {\n    this.embeddings = null;\n    this.embeddings = {};\n    // get current datetime as unix timestamp\n    let current_datetime = Math.floor(Date.now() / 1000);\n    // rename existing embeddings file to this.folder_path/embeddings-YYYY-MM-DD.json\n    await this.rename(this.file_path, this.folder_path + \"/embeddings-\" + current_datetime + \".json\");\n    // create new embeddings file\n    await this.init_embeddings_file();\n  }\n}\n\nmodule.exports = VecLite;", "const Obsidian = require(\"obsidian\");\nconst VecLite = require(\"vec-lite\");\n\nconst DEFAULT_SETTINGS = {\n  api_key: \"\",\n  chat_open: true,\n  file_exclusions: \"\",\n  folder_exclusions: \"\",\n  header_exclusions: \"\",\n  path_only: \"\",\n  show_full_path: false,\n  expanded_view: true,\n  group_nearest_by_file: false,\n  language: \"en\",\n  log_render: false,\n  log_render_files: false,\n  recently_sent_retry_notice: false,\n  skip_sections: false,\n  smart_chat_model: \"gpt-3.5-turbo-16k\",\n  view_open: true,\n  version: \"\",\n};\nconst MAX_EMBED_STRING_LENGTH = 25000;\n\nlet VERSION;\nconst SUPPORTED_FILE_TYPES = [\"md\", \"canvas\"];\n\n//create one object with all the translations\n// research : SMART_TRANSLATION[language][key]\nconst SMART_TRANSLATION = {\n  \"en\": {\n    \"pronous\": [\"my\", \"I\", \"me\", \"mine\", \"our\", \"ours\", \"us\", \"we\"],\n    \"prompt\": \"Based on your notes\",\n    \"initial_message\": \"Hi, I'm ChatGPT with access to your notes via Smart Connections. Ask me a question about your notes and I'll try to answer it.\",\n  },\n  \"es\": {\n    \"pronous\": [\"mi\", \"yo\", \"m\u00ED\", \"t\u00FA\"],\n    \"prompt\": \"Bas\u00E1ndose en sus notas\",\n    \"initial_message\": \"Hola, soy ChatGPT con acceso a tus apuntes a trav\u00E9s de Smart Connections. Hazme una pregunta sobre tus apuntes e intentar\u00E9 responderte.\",\n  },\n  \"fr\": {\n    \"pronous\": [\"me\", \"mon\", \"ma\", \"mes\", \"moi\", \"nous\", \"notre\", \"nos\", \"je\", \"j'\", \"m'\"],\n    \"prompt\": \"D'apr\u00E8s vos notes\",\n    \"initial_message\": \"Bonjour, je suis ChatGPT et j'ai acc\u00E8s \u00E0 vos notes via Smart Connections. Posez-moi une question sur vos notes et j'essaierai d'y r\u00E9pondre.\",\n  },\n  \"de\": {\n    \"pronous\": [\"mein\", \"meine\", \"meinen\", \"meiner\", \"meines\", \"mir\", \"uns\", \"unser\", \"unseren\", \"unserer\", \"unseres\"],\n    \"prompt\": \"Basierend auf Ihren Notizen\",\n    \"initial_message\": \"Hallo, ich bin ChatGPT und habe \u00FCber Smart Connections Zugang zu Ihren Notizen. Stellen Sie mir eine Frage zu Ihren Notizen und ich werde versuchen, sie zu beantworten.\",\n  },\n  \"it\": {\n    \"pronous\": [\"mio\", \"mia\", \"miei\", \"mie\", \"noi\", \"nostro\", \"nostri\", \"nostra\", \"nostre\"],\n    \"prompt\": \"Sulla base degli appunti\",\n    \"initial_message\": \"Ciao, sono ChatGPT e ho accesso ai tuoi appunti tramite Smart Connections. Fatemi una domanda sui vostri appunti e cercher\u00F2 di rispondervi.\",\n  },\n}\n\n// require built-in crypto module\nconst crypto = require(\"crypto\");\n// md5 hash using built in crypto module\nfunction md5(str) {\n  return crypto.createHash(\"md5\").update(str).digest(\"hex\");\n}\n\nclass SmartConnectionsPlugin extends Obsidian.Plugin {\n  // constructor\n  constructor() {\n    super(...arguments);\n    this.api = null;\n    this.embeddings_loaded = false;\n    this.file_exclusions = [];\n    this.folders = [];\n    this.has_new_embeddings = false;\n    this.header_exclusions = [];\n    this.nearest_cache = {};\n    this.path_only = [];\n    this.render_log = {};\n    this.render_log.deleted_embeddings = 0;\n    this.render_log.exclusions_logs = {};\n    this.render_log.failed_embeddings = [];\n    this.render_log.files = [];\n    this.render_log.new_embeddings = 0;\n    this.render_log.skipped_low_delta = {};\n    this.render_log.token_usage = 0;\n    this.render_log.tokens_saved_by_cache = 0;\n    this.retry_notice_timeout = null;\n    this.save_timeout = null;\n    this.sc_branding = {};\n    this.self_ref_kw_regex = null;\n    this.update_available = false;\n  }\n\n  async onload() {\n    // initialize when layout is ready\n    this.app.workspace.onLayoutReady(this.initialize.bind(this));\n  }\n  onunload() {\n    this.output_render_log();\n    console.log(\"unloading plugin\");\n    this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE);\n    this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE);\n  }\n  async initialize() {\n    console.log(\"Loading Smart Connections plugin\");\n    VERSION = this.manifest.version;\n    // VERSION = '1.0.0';\n    // console.log(VERSION);\n    await this.loadSettings();\n    // run after 3 seconds\n    setTimeout(this.check_for_update.bind(this), 3000);\n    // run check for update every 3 hours\n    setInterval(this.check_for_update.bind(this), 10800000);\n\n    this.addIcon();\n    this.addCommand({\n      id: \"sc-find-notes\",\n      name: \"Find: Make Smart Connections\",\n      icon: \"pencil_icon\",\n      hotkeys: [],\n      // editorCallback: async (editor) => {\n      editorCallback: async (editor) => {\n        if(editor.somethingSelected()) {\n          // get selected text\n          let selected_text = editor.getSelection();\n          // render connections from selected text\n          await this.make_connections(selected_text);\n        } else {\n          // clear nearest_cache on manual call to make connections\n          this.nearest_cache = {};\n          // console.log(\"Cleared nearest_cache\");\n          await this.make_connections();\n        }\n      }\n    });\n    this.addCommand({\n      id: \"smart-connections-view\",\n      name: \"Open: View Smart Connections\",\n      callback: () => {\n        this.open_view();\n      }\n    });\n    // open chat command\n    this.addCommand({\n      id: \"smart-connections-chat\",\n      name: \"Open: Smart Chat Conversation\",\n      callback: () => {\n        this.open_chat();\n      }\n    });\n    // open random note from nearest cache\n    this.addCommand({\n      id: \"smart-connections-random\",\n      name: \"Open: Random Note from Smart Connections\",\n      callback: () => {\n        this.open_random_note();\n      }\n    });\n    // add settings tab\n    this.addSettingTab(new SmartConnectionsSettingsTab(this.app, this));\n    // register main view type\n    this.registerView(SMART_CONNECTIONS_VIEW_TYPE, (leaf) => (new SmartConnectionsView(leaf, this)));\n    // register chat view type\n    this.registerView(SMART_CONNECTIONS_CHAT_VIEW_TYPE, (leaf) => (new SmartConnectionsChatView(leaf, this)));\n    // code-block renderer\n    this.registerMarkdownCodeBlockProcessor(\"smart-connections\", this.render_code_block.bind(this));\n\n    // if this settings.view_open is true, open view on startup\n    if(this.settings.view_open) {\n      this.open_view();\n    }\n    // if this settings.chat_open is true, open chat on startup\n    if(this.settings.chat_open) {\n      this.open_chat();\n    }\n    // on new version\n    if(this.settings.version !== VERSION) {\n      // update version\n      this.settings.version = VERSION;\n      // save settings\n      await this.saveSettings();\n      // open view\n      this.open_view();\n    }\n    // check github release endpoint if update is available\n    this.add_to_gitignore();\n    /**\n     * EXPERIMENTAL\n     * - window-based API access\n     * - code-block rendering\n     */\n    this.api = new ScSearchApi(this.app, this);\n    // register API to global window object\n    (window[\"SmartSearchApi\"] = this.api) && this.register(() => delete window[\"SmartSearchApi\"]);\n\n  }\n\n  async init_vecs() {\n    this.smart_vec_lite = new VecLite({\n      folder_path: \".smart-connections\",\n      exists_adapter: this.app.vault.adapter.exists.bind(this.app.vault.adapter),\n      mkdir_adapter: this.app.vault.adapter.mkdir.bind(this.app.vault.adapter),\n      read_adapter: this.app.vault.adapter.read.bind(this.app.vault.adapter),\n      rename_adapter: this.app.vault.adapter.rename.bind(this.app.vault.adapter),\n      stat_adapter: this.app.vault.adapter.stat.bind(this.app.vault.adapter),\n      write_adapter: this.app.vault.adapter.write.bind(this.app.vault.adapter),\n    });\n    this.embeddings_loaded = await this.smart_vec_lite.load();\n    return this.embeddings_loaded;\n  }\n  async update_to_v2() {\n    // if license key is not set, return\n    if(!this.settings.license_key) return new Obsidian.Notice(\"[Smart Connections] Supporter license key required for early access to V2\");\n    // download https://github.com/brianpetro/obsidian-smart-connections/releases/download/1.6.37/main.js\n    const v2 = await (0, Obsidian.requestUrl)({\n      url: \"https://sync.smartconnections.app/download_v2\",\n      method: \"POST\",\n      headers: {\n        \"Content-Type\": \"application/json\",\n      },\n      body: JSON.stringify({\n        license_key: this.settings.license_key,\n      })\n    });\n    if(v2.status !== 200) return console.error(\"Error downloading version 2\", v2);\n    console.log(v2);\n    await this.app.vault.adapter.write(\".obsidian/plugins/smart-connections/main.js\", v2.json.main); // add new\n    await this.app.vault.adapter.write(\".obsidian/plugins/smart-connections/manifest.json\", v2.json.manifest); // add new\n    await this.app.vault.adapter.write(\".obsidian/plugins/smart-connections/styles.css\", v2.json.styles); // add new\n    window.restart_plugin = async (id) => {\n      console.log(\"restarting plugin\", id);\n      await window.app.plugins.disablePlugin(id);\n      await window.app.plugins.enablePlugin(id);\n      console.log(\"plugin restarted\", id);\n    }\n    window.restart_plugin(this.manifest.id);\n  }\n\n  async loadSettings() {\n    this.settings = Object.assign({}, DEFAULT_SETTINGS, await this.loadData());\n    // load file exclusions if not blank\n    if(this.settings.file_exclusions && this.settings.file_exclusions.length > 0) {\n      // split file exclusions into array and trim whitespace\n      this.file_exclusions = this.settings.file_exclusions.split(\",\").map((file) => {\n        return file.trim();\n      });\n    }\n    // load folder exclusions if not blank\n    if(this.settings.folder_exclusions && this.settings.folder_exclusions.length > 0) {\n      // add slash to end of folder name if not present\n      const folder_exclusions = this.settings.folder_exclusions.split(\",\").map((folder) => {\n        // trim whitespace\n        folder = folder.trim();\n        if(folder.slice(-1) !== \"/\") {\n          return folder + \"/\";\n        } else {\n          return folder;\n        }\n      });\n      // merge folder exclusions with file exclusions\n      this.file_exclusions = this.file_exclusions.concat(folder_exclusions);\n    }\n    // load header exclusions if not blank\n    if(this.settings.header_exclusions && this.settings.header_exclusions.length > 0) {\n      this.header_exclusions = this.settings.header_exclusions.split(\",\").map((header) => {\n        return header.trim();\n      });\n    }\n    // load path_only if not blank\n    if(this.settings.path_only && this.settings.path_only.length > 0) {\n      this.path_only = this.settings.path_only.split(\",\").map((path) => {\n        return path.trim();\n      });\n    }\n    // load self_ref_kw_regex\n    this.self_ref_kw_regex = new RegExp(`\\\\b(${SMART_TRANSLATION[this.settings.language].pronous.join(\"|\")})\\\\b`, \"gi\");\n    // load failed files\n    await this.load_failed_files();\n  }\n  async saveSettings(rerender=false) {\n    await this.saveData(this.settings);\n    // re-load settings into memory\n    await this.loadSettings();\n    // re-render view if set to true (for example, after adding API key)\n    if(rerender) {\n      this.nearest_cache = {};\n      await this.make_connections();\n    }\n  }\n\n  // check for update\n  async check_for_update() {\n    // fail silently, ex. if no internet connection\n    try {\n      // get latest release version from github\n      const response = await (0, Obsidian.requestUrl)({\n        url: \"https://api.github.com/repos/brianpetro/obsidian-smart-connections/releases/latest\",\n        method: \"GET\",\n        headers: {\n          \"Content-Type\": \"application/json\",\n        },\n        contentType: \"application/json\",\n      });\n      // get version number from response\n      const latest_release = JSON.parse(response.text).tag_name;\n      // console.log(`Latest release: ${latest_release}`);\n      // if latest_release is newer than current version, show message\n      if(latest_release !== VERSION) {\n        new Obsidian.Notice(`[Smart Connections] A new version is available! (v${latest_release})`);\n        this.update_available = true;\n        this.render_brand(\"all\")\n      }\n    } catch (error) {\n      console.log(error);\n    }\n  }\n\n  async render_code_block(contents, container, ctx) {\n    let nearest;\n    if(contents.trim().length > 0) {\n      nearest = await this.api.search(contents);\n    } else {\n      // use ctx to get file\n      console.log(ctx);\n      const file = this.app.vault.getAbstractFileByPath(ctx.sourcePath);\n      nearest = await this.find_note_connections(file);\n    }\n    if (nearest.length) {\n      this.update_results(container, nearest);\n    }\n  }\n\n  async make_connections(selected_text=null) {\n    let view = this.get_view();\n    if (!view) {\n      // open view if not open\n      await this.open_view();\n      view = this.get_view();\n    }\n    await view.render_connections(selected_text);\n  }\n\n  addIcon(){\n    Obsidian.addIcon(\"smart-connections\", `<path d=\"M50,20 L80,40 L80,60 L50,100\" stroke=\"currentColor\" stroke-width=\"4\" fill=\"none\"/>\n    <path d=\"M30,50 L55,70\" stroke=\"currentColor\" stroke-width=\"5\" fill=\"none\"/>\n    <circle cx=\"50\" cy=\"20\" r=\"9\" fill=\"currentColor\"/>\n    <circle cx=\"80\" cy=\"40\" r=\"9\" fill=\"currentColor\"/>\n    <circle cx=\"80\" cy=\"70\" r=\"9\" fill=\"currentColor\"/>\n    <circle cx=\"50\" cy=\"100\" r=\"9\" fill=\"currentColor\"/>\n    <circle cx=\"30\" cy=\"50\" r=\"9\" fill=\"currentColor\"/>`);\n  }\n\n  // open random note\n  async open_random_note() {\n    const curr_file = this.app.workspace.getActiveFile();\n    const curr_key = md5(curr_file.path);\n    // if no nearest cache, create Obsidian notice\n    if(typeof this.nearest_cache[curr_key] === \"undefined\") {\n      new Obsidian.Notice(\"[Smart Connections] No Smart Connections found. Open a note to get Smart Connections.\");\n      return;\n    }\n    // get random from nearest cache\n    const rand = Math.floor(Math.random() * this.nearest_cache[curr_key].length/2); // divide by 2 to limit to top half of results\n    const random_file = this.nearest_cache[curr_key][rand];\n    // open random file\n    this.open_note(random_file);\n  }\n\n  async open_view() {\n    if(this.get_view()){\n      console.log(\"Smart Connections view already open\");\n      return;\n    }\n    this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE);\n    await this.app.workspace.getRightLeaf(false).setViewState({\n      type: SMART_CONNECTIONS_VIEW_TYPE,\n      active: true,\n    });\n    this.app.workspace.revealLeaf(\n      this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE)[0]\n    );\n  }\n  // source: https://github.com/obsidianmd/obsidian-releases/blob/master/plugin-review.md#avoid-managing-references-to-custom-views\n  get_view() {\n    for (let leaf of this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE)) {\n      if (leaf.view instanceof SmartConnectionsView) {\n        return leaf.view;\n      }\n    }\n  }\n  // open chat view\n  async open_chat(retries=0) {\n    if(!this.embeddings_loaded) {\n      console.log(\"embeddings not loaded yet\");\n      if(retries < 3) {\n        // wait and try again\n        setTimeout(() => {\n          this.open_chat(retries+1);\n        }, 1000 * (retries+1));\n        return;\n      }\n      console.log(\"embeddings still not loaded, opening smart view\");\n      this.open_view();\n      return;\n    }\n    this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE);\n    await this.app.workspace.getRightLeaf(false).setViewState({\n      type: SMART_CONNECTIONS_CHAT_VIEW_TYPE,\n      active: true,\n    });\n    this.app.workspace.revealLeaf(\n      this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE)[0]\n    );\n  }\n  \n  // get embeddings for all files\n  async get_all_embeddings() {\n    // get all files in vault and filter all but markdown and canvas files\n    const files = (await this.app.vault.getFiles()).filter((file) => file instanceof Obsidian.TFile && (file.extension === \"md\" || file.extension === \"canvas\"));\n    // const files = await this.app.vault.getMarkdownFiles();\n    // get open files to skip if file is currently open\n    const open_files = this.app.workspace.getLeavesOfType(\"markdown\").map((leaf) => leaf.view.file);\n    const clean_up_log = this.smart_vec_lite.clean_up_embeddings(files);\n    if(this.settings.log_render){\n      this.render_log.total_files = files.length;\n      this.render_log.deleted_embeddings = clean_up_log.deleted_embeddings;\n      this.render_log.total_embeddings = clean_up_log.total_embeddings;\n    }\n    // batch embeddings\n    let batch_promises = [];\n    for (let i = 0; i < files.length; i++) {\n      // skip if path contains a #\n      if(files[i].path.indexOf(\"#\") > -1) {\n        // console.log(\"skipping file '\"+files[i].path+\"' (path contains #)\");\n        this.log_exclusion(\"path contains #\");\n        continue;\n      }\n      // skip if file already has embedding and embedding.mtime is greater than or equal to file.mtime\n      if(this.smart_vec_lite.mtime_is_current(md5(files[i].path), files[i].stat.mtime)) {\n        // log skipping file\n        // console.log(\"skipping file (mtime)\");\n        continue;\n      }\n      // check if file is in failed_files\n      if(this.settings.failed_files.indexOf(files[i].path) > -1) {\n        // log skipping file\n        // console.log(\"skipping previously failed file, use button in settings to retry\");\n        // use setTimeout to prevent multiple notices\n        if(this.retry_notice_timeout) {\n          clearTimeout(this.retry_notice_timeout);\n          this.retry_notice_timeout = null;\n        }\n        // limit to one notice every 10 minutes\n        if(!this.recently_sent_retry_notice){\n          new Obsidian.Notice(\"Smart Connections: Skipping previously failed file, use button in settings to retry\");\n          this.recently_sent_retry_notice = true;\n          setTimeout(() => {\n            this.recently_sent_retry_notice = false;  \n          }, 600000);\n        }\n        continue;\n      }\n      // skip files where path contains any exclusions\n      let skip = false;\n      for(let j = 0; j < this.file_exclusions.length; j++) {\n        if(files[i].path.indexOf(this.file_exclusions[j]) > -1) {\n          skip = true;\n          this.log_exclusion(this.file_exclusions[j]);\n          // break out of loop\n          break;\n        }\n      }\n      if(skip) {\n        continue; // to next file\n      }\n      // check if file is open\n      if(open_files.indexOf(files[i]) > -1) {\n        // console.log(\"skipping file (open)\");\n        continue;\n      }\n      try {\n        // push promise to batch_promises\n        batch_promises.push(this.get_file_embeddings(files[i], false));\n      } catch (error) {\n        console.log(error);\n      }\n      // if batch_promises length is 10\n      if(batch_promises.length > 3) {\n        // wait for all promises to resolve\n        await Promise.all(batch_promises);\n        // clear batch_promises\n        batch_promises = [];\n      }\n\n      // save embeddings JSON to file every 100 files to save progress on bulk embedding\n      if(i > 0 && i % 100 === 0) {\n        await this.save_embeddings_to_file();\n      }\n    }\n    // wait for all promises to resolve\n    await Promise.all(batch_promises);\n    // write embeddings JSON to file\n    await this.save_embeddings_to_file();\n    // if render_log.failed_embeddings then update failed_embeddings.txt\n    if(this.render_log.failed_embeddings.length > 0) {\n      await this.save_failed_embeddings();\n    }\n  }\n\n  async save_embeddings_to_file(force=false) {\n    if(!this.has_new_embeddings){\n      return;\n    }\n    // console.log(\"new embeddings, saving to file\");\n    if(!force) {\n      // prevent excessive writes to embeddings file by waiting 1 minute before writing\n      if(this.save_timeout) {\n        clearTimeout(this.save_timeout);\n        this.save_timeout = null;  \n      }\n      this.save_timeout = setTimeout(() => {\n        // console.log(\"writing embeddings to file\");\n        this.save_embeddings_to_file(true);\n        // clear timeout\n        if(this.save_timeout) {\n          clearTimeout(this.save_timeout);\n          this.save_timeout = null;\n        }\n      }, 30000);\n      console.log(\"scheduled save\");\n      return;\n    }\n\n    try{\n      // use smart_vec_lite\n      await this.smart_vec_lite.save();\n      this.has_new_embeddings = false;\n    }catch(error){\n      console.log(error);\n      new Obsidian.Notice(\"Smart Connections: \"+error.message);\n    }\n\n  }\n  // save failed embeddings to file from render_log.failed_embeddings\n  async save_failed_embeddings () {\n    // write failed_embeddings to file one line per failed embedding\n    let failed_embeddings = [];\n    // if file already exists then read it\n    const failed_embeddings_file_exists = await this.app.vault.adapter.exists(\".smart-connections/failed-embeddings.txt\");\n    if(failed_embeddings_file_exists) {\n      failed_embeddings = await this.app.vault.adapter.read(\".smart-connections/failed-embeddings.txt\");\n      // split failed_embeddings into array\n      failed_embeddings = failed_embeddings.split(\"\\r\\n\");\n    }\n    // merge failed_embeddings with render_log.failed_embeddings\n    failed_embeddings = failed_embeddings.concat(this.render_log.failed_embeddings);\n    // remove duplicates\n    failed_embeddings = [...new Set(failed_embeddings)];\n    // sort failed_embeddings array alphabetically\n    failed_embeddings.sort();\n    // convert failed_embeddings array to string\n    failed_embeddings = failed_embeddings.join(\"\\r\\n\");\n    // write failed_embeddings to file\n    await this.app.vault.adapter.write(\".smart-connections/failed-embeddings.txt\", failed_embeddings);\n    // reload failed_embeddings to prevent retrying failed files until explicitly requested\n    await this.load_failed_files();\n  }\n  \n  // load failed files from failed-embeddings.txt\n  async load_failed_files () {\n    // check if failed-embeddings.txt exists\n    const failed_embeddings_file_exists = await this.app.vault.adapter.exists(\".smart-connections/failed-embeddings.txt\");\n    if(!failed_embeddings_file_exists) {\n      this.settings.failed_files = [];\n      console.log(\"No failed files.\");\n      return;\n    }\n    // read failed-embeddings.txt\n    const failed_embeddings = await this.app.vault.adapter.read(\".smart-connections/failed-embeddings.txt\");\n    // split failed_embeddings into array and remove empty lines\n    const failed_embeddings_array = failed_embeddings.split(\"\\r\\n\");\n    // split at '#' and reduce into unique file paths\n    const failed_files = failed_embeddings_array.map(embedding => embedding.split(\"#\")[0]).reduce((unique, item) => unique.includes(item) ? unique : [...unique, item], []);\n    // return failed_files\n    this.settings.failed_files = failed_files;\n    // console.log(failed_files);\n  }\n  // retry failed embeddings\n  async retry_failed_files () {\n    // remove failed files from failed_files\n    this.settings.failed_files = [];\n    // if failed-embeddings.txt exists then delete it\n    const failed_embeddings_file_exists = await this.app.vault.adapter.exists(\".smart-connections/failed-embeddings.txt\");\n    if(failed_embeddings_file_exists) {\n      await this.app.vault.adapter.remove(\".smart-connections/failed-embeddings.txt\");\n    }\n    // run get all embeddings\n    await this.get_all_embeddings();\n  }\n\n\n  // add .smart-connections to .gitignore to prevent issues with large, frequently updated embeddings file(s)\n  async add_to_gitignore() {\n    if(!(await this.app.vault.adapter.exists(\".gitignore\"))) {\n      return; // if .gitignore doesn't exist then don't add .smart-connections to .gitignore\n    }\n    let gitignore_file = await this.app.vault.adapter.read(\".gitignore\");\n    // if .smart-connections not in .gitignore\n    if (gitignore_file.indexOf(\".smart-connections\") < 0) {\n      // add .smart-connections to .gitignore\n      let add_to_gitignore = \"\\n\\n# Ignore Smart Connections folder because embeddings file is large and updated frequently\";\n      add_to_gitignore += \"\\n.smart-connections\";\n      await this.app.vault.adapter.write(\".gitignore\", gitignore_file + add_to_gitignore);\n      console.log(\"added .smart-connections to .gitignore\");\n    }\n  }\n\n  // force refresh embeddings file but first rename existing embeddings file to .smart-connections/embeddings-YYYY-MM-DD.json\n  async force_refresh_embeddings_file() {\n    new Obsidian.Notice(\"Smart Connections: embeddings file Force Refreshed, making new connections...\");\n    // force refresh\n    await this.smart_vec_lite.force_refresh();\n    // trigger making new connections\n    await this.get_all_embeddings();\n    this.output_render_log();\n    new Obsidian.Notice(\"Smart Connections: embeddings file Force Refreshed, new connections made.\");\n  }\n\n  // get embeddings for embed_input\n  async get_file_embeddings(curr_file, save=true) {\n    // let batch_promises = [];\n    let req_batch = [];\n    let blocks = [];\n    // initiate curr_file_key from md5(curr_file.path)\n    const curr_file_key = md5(curr_file.path);\n    // intiate file_file_embed_input by removing .md and converting file path to breadcrumbs (\" > \")\n    let file_embed_input = curr_file.path.replace(\".md\", \"\");\n    file_embed_input = file_embed_input.replace(/\\//g, \" > \");\n    // embed on file.name/title only if path_only path matcher specified in settings\n    let path_only = false;\n    for(let j = 0; j < this.path_only.length; j++) {\n      if(curr_file.path.indexOf(this.path_only[j]) > -1) {\n        path_only = true;\n        console.log(\"title only file with matcher: \" + this.path_only[j]);\n        // break out of loop\n        break;\n      }\n    }\n    // return early if path_only\n    if(path_only) {\n      req_batch.push([curr_file_key, file_embed_input, {\n        mtime: curr_file.stat.mtime,\n        path: curr_file.path,\n      }]);\n      await this.get_embeddings_batch(req_batch);\n      return;\n    }\n    /**\n     * BEGIN Canvas file type Embedding\n     */\n    if(curr_file.extension === \"canvas\") {\n      // get file contents and parse as JSON\n      const canvas_contents = await this.app.vault.cachedRead(curr_file);\n      if((typeof canvas_contents === \"string\") && (canvas_contents.indexOf(\"nodes\") > -1)) {\n        const canvas_json = JSON.parse(canvas_contents);\n        // for each object in nodes array\n        for(let j = 0; j < canvas_json.nodes.length; j++) {\n          // if object has text property\n          if(canvas_json.nodes[j].text) {\n            // add to file_embed_input\n            file_embed_input += \"\\n\" + canvas_json.nodes[j].text;\n          }\n          // if object has file property\n          if(canvas_json.nodes[j].file) {\n            // add to file_embed_input\n            file_embed_input += \"\\nLink: \" + canvas_json.nodes[j].file;\n          }\n        }\n      }\n      // console.log(file_embed_input);\n      req_batch.push([curr_file_key, file_embed_input, {\n        mtime: curr_file.stat.mtime,\n        path: curr_file.path,\n      }]);\n      await this.get_embeddings_batch(req_batch);\n      return;\n    }\n    \n    /**\n     * BEGIN Block \"section\" embedding\n     */\n    // get file contents\n    const note_contents = await this.app.vault.cachedRead(curr_file);\n    let processed_since_last_save = 0;\n    const note_sections = this.block_parser(note_contents, curr_file.path);\n    // console.log(note_sections);\n    // if note has more than one section (if only one then its same as full-content)\n    if(note_sections.length > 1) {\n      // for each section in file\n      //console.log(\"Sections: \" + note_sections.length);\n      for (let j = 0; j < note_sections.length; j++) {\n        // get embed_input for block\n        const block_embed_input = note_sections[j].text;\n        // console.log(note_sections[j].path);\n        // get block key from block.path (contains both file.path and header path)\n        const block_key = md5(note_sections[j].path);\n        blocks.push(block_key);\n        // skip if length of block_embed_input same as length of embeddings[block_key].meta.size\n        // TODO consider rounding to nearest 10 or 100 for fuzzy matching\n        if (this.smart_vec_lite.get_size(block_key) === block_embed_input.length) {\n          // log skipping file\n          // console.log(\"skipping block (len)\");\n          continue;\n        }\n        // add hash to blocks to prevent empty blocks triggering full-file embedding\n        // skip if embeddings key already exists and block mtime is greater than or equal to file mtime\n        if(this.smart_vec_lite.mtime_is_current(block_key, curr_file.stat.mtime)) {\n          // log skipping file\n          // console.log(\"skipping block (mtime)\");\n          continue;\n        }\n        // skip if hash is present in embeddings and hash of block_embed_input is equal to hash in embeddings\n        const block_hash = md5(block_embed_input.trim());\n        if(this.smart_vec_lite.get_hash(block_key) === block_hash) {\n          // log skipping file\n          // console.log(\"skipping block (hash)\");\n          continue;\n        }\n\n        // create req_batch for batching requests\n        req_batch.push([block_key, block_embed_input, {\n          // oldmtime: curr_file.stat.mtime, \n          // get current datetime as unix timestamp\n          mtime: Date.now(),\n          hash: block_hash, \n          parent: curr_file_key,\n          path: note_sections[j].path,\n          size: block_embed_input.length,\n        }]);\n        if(req_batch.length > 9) {\n          // add batch to batch_promises\n          await this.get_embeddings_batch(req_batch);\n          processed_since_last_save += req_batch.length;\n          // log embedding\n          // console.log(\"embedding: \" + curr_file.path);\n          if (processed_since_last_save >= 30) {\n            // write embeddings JSON to file\n            await this.save_embeddings_to_file();\n            // reset processed_since_last_save\n            processed_since_last_save = 0;\n          }\n          // reset req_batch\n          req_batch = [];\n        }\n      }\n    }\n    // if req_batch is not empty\n    if(req_batch.length > 0) {\n      // process remaining req_batch\n      await this.get_embeddings_batch(req_batch);\n      req_batch = [];\n      processed_since_last_save += req_batch.length;\n    }\n    \n    /**\n     * BEGIN File \"full note\" embedding\n     */\n\n    // if file length is less than ~8000 tokens use full file contents\n    // else if file length is greater than 8000 tokens build file_embed_input from file headings\n    file_embed_input += `:\\n`;\n    /**\n     * TODO: improve/refactor the following \"large file reduce to headings\" logic\n     */\n    if(note_contents.length < MAX_EMBED_STRING_LENGTH) {\n      file_embed_input += note_contents\n    }else{ \n      const note_meta_cache = this.app.metadataCache.getFileCache(curr_file);\n      // for each heading in file\n      if(typeof note_meta_cache.headings === \"undefined\") {\n        // console.log(\"no headings found, using first chunk of file instead\");\n        file_embed_input += note_contents.substring(0, MAX_EMBED_STRING_LENGTH);\n      }else{\n        let note_headings = \"\";\n        for (let j = 0; j < note_meta_cache.headings.length; j++) {\n          // get heading level\n          const heading_level = note_meta_cache.headings[j].level;\n          // get heading text\n          const heading_text = note_meta_cache.headings[j].heading;\n          // build markdown heading\n          let md_heading = \"\";\n          for (let k = 0; k < heading_level; k++) {\n            md_heading += \"#\";\n          }\n          // add heading to note_headings\n          note_headings += `${md_heading} ${heading_text}\\n`;\n        }\n        //console.log(note_headings);\n        file_embed_input += note_headings\n        if(file_embed_input.length > MAX_EMBED_STRING_LENGTH) {\n          file_embed_input = file_embed_input.substring(0, MAX_EMBED_STRING_LENGTH);\n        }\n      }\n    }\n    // skip embedding full file if blocks is not empty and all hashes are present in embeddings\n    // better than hashing file_embed_input because more resilient to inconsequential changes (whitespace between headings)\n    const file_hash = md5(file_embed_input.trim());\n    const existing_hash = this.smart_vec_lite.get_hash(curr_file_key);\n    if(existing_hash && (file_hash === existing_hash)) {\n      // console.log(\"skipping file (hash): \" + curr_file.path);\n      this.update_render_log(blocks, file_embed_input);\n      return;\n    };\n\n    // if not already skipping and blocks are present\n    const existing_blocks = this.smart_vec_lite.get_children(curr_file_key);\n    let existing_has_all_blocks = true;\n    if(existing_blocks && Array.isArray(existing_blocks) && (blocks.length > 0)) {\n      // if all blocks are in existing_blocks then skip (allows deletion of small blocks without triggering full file embedding)\n      for (let j = 0; j < blocks.length; j++) {\n        if(existing_blocks.indexOf(blocks[j]) === -1) {\n          existing_has_all_blocks = false;\n          break;\n        }\n      }\n    }\n    // if existing has all blocks then check file size for delta\n    if(existing_has_all_blocks){\n      // get current note file size\n      const curr_file_size = curr_file.stat.size;\n      // get file size from embeddings\n      const prev_file_size = this.smart_vec_lite.get_size(curr_file_key);\n      if (prev_file_size) {\n        // if curr file size is less than 10% different from prev file size\n        const file_delta_pct = Math.round((Math.abs(curr_file_size - prev_file_size) / curr_file_size) * 100);\n        if(file_delta_pct < 10) {\n          // skip embedding\n          // console.log(\"skipping file (size) \" + curr_file.path);\n          this.render_log.skipped_low_delta[curr_file.name] = file_delta_pct + \"%\";\n          this.update_render_log(blocks, file_embed_input);\n          return;\n        }\n      }\n    }\n    let meta = {\n      mtime: curr_file.stat.mtime,\n      hash: file_hash,\n      path: curr_file.path,\n      size: curr_file.stat.size,\n      children: blocks,\n    };\n    // batch_promises.push(this.get_embeddings(curr_file_key, file_embed_input, meta));\n    req_batch.push([curr_file_key, file_embed_input, meta]);\n    // send batch request\n    await this.get_embeddings_batch(req_batch);\n\n    // log embedding\n    // console.log(\"embedding: \" + curr_file.path);\n    if (save) {\n      // write embeddings JSON to file\n      await this.save_embeddings_to_file();\n    }\n\n  }\n\n  update_render_log(blocks, file_embed_input) {\n    if (blocks.length > 0) {\n      // multiply by 2 because implies we saved token spending on blocks(sections), too\n      this.render_log.tokens_saved_by_cache += file_embed_input.length / 2;\n    } else {\n      // calc tokens saved by cache: divide by 4 for token estimate\n      this.render_log.tokens_saved_by_cache += file_embed_input.length / 4;\n    }\n  }\n\n  async get_embeddings_batch(req_batch) {\n    console.log(\"get_embeddings_batch\");\n    // if req_batch is empty then return\n    if(req_batch.length === 0) return;\n    // create arrary of embed_inputs from req_batch[i][1]\n    const embed_inputs = req_batch.map((req) => req[1]);\n    // request embeddings from embed_inputs\n    const requestResults = await this.request_embedding_from_input(embed_inputs);\n    // if requestResults is null then return\n    if(!requestResults) {\n      console.log(\"failed embedding batch\");\n      // log failed file names to render_log\n      this.render_log.failed_embeddings = [...this.render_log.failed_embeddings, ...req_batch.map((req) => req[2].path)];\n      return;\n    }\n    // if requestResults is not null\n    if(requestResults){\n      this.has_new_embeddings = true;\n      // add embedding key to render_log\n      if(this.settings.log_render){\n        if(this.settings.log_render_files){\n          this.render_log.files = [...this.render_log.files, ...req_batch.map((req) => req[2].path)];\n        }\n        this.render_log.new_embeddings += req_batch.length;\n        // add token usage to render_log\n        this.render_log.token_usage += requestResults.usage.total_tokens;\n      }\n      // console.log(requestResults.data.length);\n      // loop through requestResults.data\n      for(let i = 0; i < requestResults.data.length; i++) {\n        const vec = requestResults.data[i].embedding;\n        const index = requestResults.data[i].index;\n        if(vec) {\n          const key = req_batch[index][0];\n          const meta = req_batch[index][2];\n          this.smart_vec_lite.save_embedding(key, vec, meta);\n        }\n      }\n    }\n  }\n\n  async request_embedding_from_input(embed_input, retries = 0) {\n    // (FOR TESTING) test fail process by forcing fail\n    // return null;\n    // check if embed_input is a string\n    // if(typeof embed_input !== \"string\") {\n    //   console.log(\"embed_input is not a string\");\n    //   return null;\n    // }\n    // check if embed_input is empty\n    if(embed_input.length === 0) {\n      console.log(\"embed_input is empty\");\n      return null;\n    }\n    const usedParams = {\n      model: \"text-embedding-ada-002\",\n      input: embed_input,\n    };\n    // console.log(this.settings.api_key);\n    const reqParams = {\n      url: `https://api.openai.com/v1/embeddings`,\n      method: \"POST\",\n      body: JSON.stringify(usedParams),\n      headers: {\n        \"Content-Type\": \"application/json\",\n        \"Authorization\": `Bearer ${this.settings.api_key}`\n      }\n    };\n    let resp;\n    try {\n      resp = await (0, Obsidian.request)(reqParams)\n      return JSON.parse(resp);\n    } catch (error) {\n      // retry request if error is 429\n      if((error.status === 429) && (retries < 3)) {\n        retries++;\n        // exponential backoff\n        const backoff = Math.pow(retries, 2);\n        console.log(`retrying request (429) in ${backoff} seconds...`);\n        await new Promise(r => setTimeout(r, 1000 * backoff));\n        return await this.request_embedding_from_input(embed_input, retries);\n      }\n      // log full error to console\n      console.log(resp);\n      // console.log(\"first line of embed: \" + embed_input.substring(0, embed_input.indexOf(\"\\n\")));\n      // console.log(\"embed input length: \"+ embed_input.length);\n      // if(Array.isArray(embed_input)) {\n      //   console.log(embed_input.map((input) => input.length));\n      // }\n      // console.log(\"erroneous embed input: \" + embed_input);\n      console.log(error);\n      // console.log(usedParams);\n      // console.log(usedParams.input.length);\n      return null; \n    }\n  }\n  async test_api_key() {\n    const embed_input = \"This is a test of the OpenAI API.\";\n    const resp = await this.request_embedding_from_input(embed_input);\n    if(resp && resp.usage) {\n      console.log(\"API key is valid\");\n      return true;\n    }else{\n      console.log(\"API key is invalid\");\n      return false;\n    }\n  }\n\n\n  output_render_log() {\n    // if settings.log_render is true\n    if(this.settings.log_render) {\n      if (this.render_log.new_embeddings === 0) {\n        return;\n      }else{\n        // pretty print this.render_log to console\n        console.log(JSON.stringify(this.render_log, null, 2));\n      }\n    }\n\n    // clear render_log\n    this.render_log = {};\n    this.render_log.deleted_embeddings = 0;\n    this.render_log.exclusions_logs = {};\n    this.render_log.failed_embeddings = [];\n    this.render_log.files = [];\n    this.render_log.new_embeddings = 0;\n    this.render_log.skipped_low_delta = {};\n    this.render_log.token_usage = 0;\n    this.render_log.tokens_saved_by_cache = 0;\n  }\n\n  // find connections by most similar to current note by cosine similarity\n  async find_note_connections(current_note=null) {\n    // md5 of current note path\n    const curr_key = md5(current_note.path);\n    // if in this.nearest_cache then set to nearest\n    // else get nearest\n    let nearest = [];\n    if(this.nearest_cache[curr_key]) {\n      nearest = this.nearest_cache[curr_key];\n      // console.log(\"nearest from cache\");\n    }else{\n      // skip files where path contains any exclusions\n      for(let j = 0; j < this.file_exclusions.length; j++) {\n        if(current_note.path.indexOf(this.file_exclusions[j]) > -1) {\n          this.log_exclusion(this.file_exclusions[j]);\n          // break out of loop and finish here\n          return \"excluded\";\n        }\n      }\n      // get all embeddings\n      // await this.get_all_embeddings();\n      // wrap get all in setTimeout to allow for UI to update\n      setTimeout(() => {\n        this.get_all_embeddings()\n      }, 3000);\n      // get from cache if mtime is same and values are not empty\n      if(this.smart_vec_lite.mtime_is_current(curr_key, current_note.stat.mtime)) {\n        // skipping get file embeddings because nothing has changed\n        // console.log(\"find_note_connections - skipping file (mtime)\");\n      }else{\n        // get file embeddings\n        await this.get_file_embeddings(current_note);\n      }\n      // get current note embedding vector\n      const vec = this.smart_vec_lite.get_vec(curr_key);\n      if(!vec) {\n        return \"Error getting embeddings for: \"+current_note.path;\n      }\n      \n      // compute cosine similarity between current note and all other notes via embeddings\n      nearest = this.smart_vec_lite.nearest(vec, {\n        skip_key: curr_key,\n        skip_sections: this.settings.skip_sections,\n      });\n  \n      // save to this.nearest_cache\n      this.nearest_cache[curr_key] = nearest;\n    }\n\n    // return array sorted by cosine similarity\n    return nearest;\n  }\n  \n  // create render_log object of exlusions with number of times skipped as value\n  log_exclusion(exclusion) {\n    // increment render_log for skipped file\n    this.render_log.exclusions_logs[exclusion] = (this.render_log.exclusions_logs[exclusion] || 0) + 1;\n  }\n  \n\n  block_parser(markdown, file_path){\n    // if this.settings.skip_sections is true then return empty array\n    if(this.settings.skip_sections) {\n      return [];\n    }\n    // split the markdown into lines\n    const lines = markdown.split('\\n');\n    // initialize the blocks array\n    let blocks = [];\n    // current headers array\n    let currentHeaders = [];\n    // remove .md file extension and convert file_path to breadcrumb formatting\n    const file_breadcrumbs = file_path.replace('.md', '').replace(/\\//g, ' > ');\n    // initialize the block string\n    let block = '';\n    let block_headings = '';\n    let block_path = file_path;\n\n    let last_heading_line = 0;\n    let i = 0;\n    let block_headings_list = [];\n    // loop through the lines\n    for (i = 0; i < lines.length; i++) {\n      // get the line\n      const line = lines[i];\n      // if line does not start with #\n      // or if line starts with # and second character is a word or number indicating a \"tag\"\n      // then add to block\n      if (!line.startsWith('#') || (['#',' '].indexOf(line[1]) < 0)){\n        // skip if line is empty\n        if(line === '') continue;\n        // skip if line is empty bullet or checkbox\n        if(['- ', '- [ ] '].indexOf(line) > -1) continue;\n        // if currentHeaders is empty skip (only blocks with headers, otherwise block.path conflicts with file.path)\n        if(currentHeaders.length === 0) continue;\n        // add line to block\n        block += \"\\n\" + line;\n        continue;\n      }\n      /**\n       * BEGIN Heading parsing\n       * - likely a heading if made it this far\n       */\n      last_heading_line = i;\n      // push the current block to the blocks array unless last line was a also a header\n      if(i > 0 && (last_heading_line !== (i-1)) && (block.indexOf(\"\\n\") > -1) && this.validate_headings(block_headings)) {\n        output_block();\n      }\n      // get the header level\n      const level = line.split('#').length - 1;\n      // remove any headers from the current headers array that are higher than the current header level\n      currentHeaders = currentHeaders.filter(header => header.level < level);\n      // add header and level to current headers array\n      // trim the header to remove \"#\" and any trailing spaces\n      currentHeaders.push({header: line.replace(/#/g, '').trim(), level: level});\n      // initialize the block breadcrumbs with file.path the current headers\n      block = file_breadcrumbs;\n      block += \": \" + currentHeaders.map(header => header.header).join(' > ');\n      block_headings = \"#\"+currentHeaders.map(header => header.header).join('#');\n      // if block_headings is already in block_headings_list then add a number to the end\n      if(block_headings_list.indexOf(block_headings) > -1) {\n        let count = 1;\n        while(block_headings_list.indexOf(`${block_headings}{${count}}`) > -1) {\n          count++;\n        }\n        block_headings = `${block_headings}{${count}}`;\n      }\n      block_headings_list.push(block_headings);\n      block_path = file_path + block_headings;\n    }\n    // handle remaining after loop\n    if((last_heading_line !== (i-1)) && (block.indexOf(\"\\n\") > -1) && this.validate_headings(block_headings)) output_block();\n    // remove any blocks that are too short (length < 50)\n    blocks = blocks.filter(b => b.length > 50);\n    // console.log(blocks);\n    // return the blocks array\n    return blocks;\n\n    function output_block() {\n      // breadcrumbs length (first line of block)\n      const breadcrumbs_length = block.indexOf(\"\\n\") + 1;\n      const block_length = block.length - breadcrumbs_length;\n      // trim block to max length\n      if (block.length > MAX_EMBED_STRING_LENGTH) {\n        block = block.substring(0, MAX_EMBED_STRING_LENGTH);\n      }\n      blocks.push({ text: block.trim(), path: block_path, length: block_length });\n    }\n  }\n  // reverse-retrieve block given path\n  async block_retriever(path, limits={}) {\n    limits = {\n      lines: null,\n      chars_per_line: null,\n      max_chars: null,\n      ...limits\n    }\n    // return if no # in path\n    if (path.indexOf('#') < 0) {\n      console.log(\"not a block path: \"+path);\n      return false;\n    }\n    let block = [];\n    let block_headings = path.split('#').slice(1);\n    // if path ends with number in curly braces\n    let heading_occurrence = 0;\n    if(block_headings[block_headings.length-1].indexOf('{') > -1) {\n      // get the occurrence number\n      heading_occurrence = parseInt(block_headings[block_headings.length-1].split('{')[1].replace('}', ''));\n      // remove the occurrence from the last heading\n      block_headings[block_headings.length-1] = block_headings[block_headings.length-1].split('{')[0];\n    }\n    let currentHeaders = [];\n    let occurrence_count = 0;\n    let begin_line = 0;\n    let i = 0;\n    // get file path from path\n    const file_path = path.split('#')[0];\n    // get file\n    const file = this.app.vault.getAbstractFileByPath(file_path);\n    if(!(file instanceof Obsidian.TFile)) {\n      console.log(\"not a file: \"+file_path);\n      return false;\n    }\n    // get file contents\n    const file_contents = await this.app.vault.cachedRead(file);\n    // split the file contents into lines\n    const lines = file_contents.split('\\n');\n    // loop through the lines\n    let is_code = false;\n    for (i = 0; i < lines.length; i++) {\n      // get the line\n      const line = lines[i];\n      // if line begins with three backticks then toggle is_code\n      if(line.indexOf('```') === 0) {\n        is_code = !is_code;\n      }\n      // if is_code is true then add line with preceding tab and continue\n      if(is_code) {\n        continue;\n      }\n      // skip if line is empty bullet or checkbox\n      if(['- ', '- [ ] '].indexOf(line) > -1) continue;\n      // if line does not start with #\n      // or if line starts with # and second character is a word or number indicating a \"tag\"\n      // then continue to next line\n      if (!line.startsWith('#') || (['#',' '].indexOf(line[1]) < 0)){\n        continue;\n      }\n      /**\n       * BEGIN Heading parsing\n       * - likely a heading if made it this far\n       */\n      // get the heading text\n      const heading_text = line.replace(/#/g, '').trim();\n      // continue if heading text is not in block_headings\n      const heading_index = block_headings.indexOf(heading_text);\n      if (heading_index < 0) continue;\n      // if currentHeaders.length !== heading_index then we have a mismatch\n      if (currentHeaders.length !== heading_index) continue;\n      // push the heading text to the currentHeaders array\n      currentHeaders.push(heading_text);\n      // if currentHeaders.length === block_headings.length then we have a match\n      if (currentHeaders.length === block_headings.length) {\n        // if heading_occurrence is defined then increment occurrence_count\n        if(heading_occurrence === 0) {\n          // set begin_line to i + 1\n          begin_line = i + 1;\n          break; // break out of loop\n        }\n        // if occurrence_count !== heading_occurrence then continue\n        if(occurrence_count === heading_occurrence){\n          begin_line = i + 1;\n          break; // break out of loop\n        }\n        occurrence_count++;\n        // reset currentHeaders\n        currentHeaders.pop();\n        continue;\n      }\n    }\n    // if no begin_line then return false\n    if (begin_line === 0) return false;\n    // iterate through lines starting at begin_line\n    is_code = false;\n    // character accumulator\n    let char_count = 0;\n    for (i = begin_line; i < lines.length; i++) {\n      if((typeof line_limit === \"number\") && (block.length > line_limit)){\n        block.push(\"...\");\n        break; // ends when line_limit is reached\n      }\n      let line = lines[i];\n      if ((line.indexOf('#') === 0) && (['#',' '].indexOf(line[1]) !== -1)){\n        break; // ends when encountering next header\n      }\n      // DEPRECATED: should be handled by new_line+char_count check (happens in previous iteration)\n      // if char_count is greater than limit.max_chars, skip\n      if (limits.max_chars && char_count > limits.max_chars) {\n        block.push(\"...\");\n        break;\n      }\n      // if new_line + char_count is greater than limit.max_chars, skip\n      if (limits.max_chars && ((line.length + char_count) > limits.max_chars)) {\n        const max_new_chars = limits.max_chars - char_count;\n        line = line.slice(0, max_new_chars) + \"...\";\n        break;\n      }\n      // validate/format\n      // if line is empty, skip\n      if (line.length === 0) continue;\n      // limit length of line to N characters\n      if (limits.chars_per_line && line.length > limits.chars_per_line) {\n        line = line.slice(0, limits.chars_per_line) + \"...\";\n      }\n      // if line is a code block, skip\n      if (line.startsWith(\"```\")) {\n        is_code = !is_code;\n        continue;\n      }\n      if (is_code){\n        // add tab to beginning of line\n        line = \"\\t\"+line;\n      }\n      // add line to block\n      block.push(line);\n      // increment char_count\n      char_count += line.length;\n    }\n    // close code block if open\n    if (is_code) {\n      block.push(\"```\");\n    }\n    return block.join(\"\\n\").trim();\n  }\n\n  // retrieve a file from the vault\n  async file_retriever(link, limits={}) {\n    limits = {\n      lines: null,\n      max_chars: null,\n      chars_per_line: null,\n      ...limits\n    };\n    const this_file = this.app.vault.getAbstractFileByPath(link);\n    // if file is not found, skip\n    if (!(this_file instanceof Obsidian.TAbstractFile)) return false;\n    // use cachedRead to get the first 10 lines of the file\n    const file_content = await this.app.vault.cachedRead(this_file);\n    const file_lines = file_content.split(\"\\n\");\n    let first_ten_lines = [];\n    let is_code = false;\n    let char_accum = 0;\n    const line_limit = limits.lines || file_lines.length;\n    for (let i = 0; first_ten_lines.length < line_limit; i++) {\n      let line = file_lines[i];\n      // if line is undefined, break\n      if (typeof line === 'undefined')\n        break;\n      // if line is empty, skip\n      if (line.length === 0)\n        continue;\n      // limit length of line to N characters\n      if (limits.chars_per_line && line.length > limits.chars_per_line) {\n        line = line.slice(0, limits.chars_per_line) + \"...\";\n      }\n      // if line is \"---\", skip\n      if (line === \"---\")\n        continue;\n      // skip if line is empty bullet or checkbox\n      if (['- ', '- [ ] '].indexOf(line) > -1)\n        continue;\n      // if line is a code block, skip\n      if (line.indexOf(\"```\") === 0) {\n        is_code = !is_code;\n        continue;\n      }\n      // if char_accum is greater than limit.max_chars, skip\n      if (limits.max_chars && char_accum > limits.max_chars) {\n        first_ten_lines.push(\"...\");\n        break;\n      }\n      if (is_code) {\n        // if is code, add tab to beginning of line\n        line = \"\\t\" + line;\n      }\n      // if line is a heading\n      if (line_is_heading(line)) {\n        // look at last line in first_ten_lines to see if it is a heading\n        // note: uses last in first_ten_lines, instead of look ahead in file_lines, because..\n        // ...next line may be excluded from first_ten_lines by previous if statements\n        if ((first_ten_lines.length > 0) && line_is_heading(first_ten_lines[first_ten_lines.length - 1])) {\n          // if last line is a heading, remove it\n          first_ten_lines.pop();\n        }\n      }\n      // add line to first_ten_lines\n      first_ten_lines.push(line);\n      // increment char_accum\n      char_accum += line.length;\n    }\n    // for each line in first_ten_lines, apply view-specific formatting\n    for (let i = 0; i < first_ten_lines.length; i++) {\n      // if line is a heading\n      if (line_is_heading(first_ten_lines[i])) {\n        // if this is the last line in first_ten_lines\n        if (i === first_ten_lines.length - 1) {\n          // remove the last line if it is a heading\n          first_ten_lines.pop();\n          break;\n        }\n        // remove heading syntax to improve readability in small space\n        first_ten_lines[i] = first_ten_lines[i].replace(/#+/, \"\");\n        first_ten_lines[i] = `\\n${first_ten_lines[i]}:`;\n      }\n    }\n    // join first ten lines into string\n    first_ten_lines = first_ten_lines.join(\"\\n\");\n    return first_ten_lines;\n  }\n\n  // iterate through blocks and skip if block_headings contains this.header_exclusions\n  validate_headings(block_headings) {\n    let valid = true;\n    if (this.header_exclusions.length > 0) {\n      for (let k = 0; k < this.header_exclusions.length; k++) {\n        if (block_headings.indexOf(this.header_exclusions[k]) > -1) {\n          valid = false;\n          this.log_exclusion(\"heading: \"+this.header_exclusions[k]);\n          break;\n        }\n      }\n    }\n    return valid;\n  }\n  // render \"Smart Connections\" text fixed in the bottom right corner\n  render_brand(container, location=\"default\") {\n    // if location is all then get Object.keys(this.sc_branding) and call this function for each\n    if (container === \"all\") {\n      const locations = Object.keys(this.sc_branding);\n      for (let i = 0; i < locations.length; i++) {\n        this.render_brand(this.sc_branding[locations[i]], locations[i]);\n      }\n      return;\n    }\n    // brand container\n    this.sc_branding[location] = container;\n    // if this.sc_branding[location] contains child with class \"sc-brand\", remove it\n    if (this.sc_branding[location].querySelector(\".sc-brand\")) {\n      this.sc_branding[location].querySelector(\".sc-brand\").remove();\n    }\n    const brand_container = this.sc_branding[location].createEl(\"div\", { cls: \"sc-brand\" });\n    // add text\n    // add SVG signal icon using getIcon\n    Obsidian.setIcon(brand_container, \"smart-connections\");\n    const brand_p = brand_container.createEl(\"p\");\n    let text = \"Smart Connections\";\n    let attr = {};\n    // if update available, change text to \"Update Available\"\n    if (this.update_available) {\n      text = \"Update Available\";\n      attr = {\n        style: \"font-weight: 700;\"\n      };\n    }\n    brand_p.createEl(\"a\", {\n      cls: \"\",\n      text: text,\n      href: \"https://github.com/brianpetro/obsidian-smart-connections/discussions\",\n      target: \"_blank\",\n      attr: attr\n    });\n  }\n\n\n  // create list of nearest notes\n  async update_results(container, nearest) {\n    let list;\n    // check if list exists\n    if((container.children.length > 1) && (container.children[1].classList.contains(\"sc-list\"))){\n      list = container.children[1];\n    }\n    // if list exists, empty it\n    if (list) {\n      list.empty();\n    } else {\n      // create list element\n      list = container.createEl(\"div\", { cls: \"sc-list\" });\n    }\n    let search_result_class = \"search-result\";\n    // if settings expanded_view is false, add sc-collapsed class\n    if(!this.settings.expanded_view) search_result_class += \" sc-collapsed\";\n\n    // TODO: add option to group nearest by file\n    if(!this.settings.group_nearest_by_file) {\n      // for each nearest note\n      for (let i = 0; i < nearest.length; i++) {\n        /**\n         * BEGIN EXTERNAL LINK LOGIC\n         * if link is an object, it indicates external link\n         */\n        if (typeof nearest[i].link === \"object\") {\n          const item = list.createEl(\"div\", { cls: \"search-result\" });\n          const link = item.createEl(\"a\", {\n            cls: \"search-result-file-title is-clickable\",\n            href: nearest[i].link.path,\n            title: nearest[i].link.title,\n          });\n          link.innerHTML = this.render_external_link_elm(nearest[i].link);\n          item.setAttr('draggable', 'true')\n          continue; // ends here for external links\n        }\n        /**\n         * BEGIN INTERNAL LINK LOGIC\n         * if link is a string, it indicates internal link\n         */\n        let file_link_text;\n        const file_similarity_pct = Math.round(nearest[i].similarity * 100) + \"%\";\n        if(this.settings.show_full_path) {\n          const pcs = nearest[i].link.split(\"/\");\n          file_link_text = pcs[pcs.length - 1];\n          const path = pcs.slice(0, pcs.length - 1).join(\"/\");\n          // file_link_text = `<small>${path} | ${file_similarity_pct}</small><br>${file_link_text}`;\n          file_link_text = `<small>${file_similarity_pct} | ${path} | ${file_link_text}</small>`;\n        }else{\n          file_link_text = '<small>' + file_similarity_pct + \" | \" + nearest[i].link.split(\"/\").pop() + '</small>';\n        }\n        // skip contents rendering if incompatible file type\n        // ex. not markdown file or contains no '.excalidraw'\n        if(!this.renderable_file_type(nearest[i].link)){\n          const item = list.createEl(\"div\", { cls: \"search-result\" });\n          const link = item.createEl(\"a\", {\n            cls: \"search-result-file-title is-clickable\",\n            href: nearest[i].link,\n          });\n          link.innerHTML = file_link_text;\n          // drag and drop\n          item.setAttr('draggable', 'true')\n          // add listeners to link\n          this.add_link_listeners(link, nearest[i], item);\n          continue;\n        }\n\n        // remove file extension if .md and make # into >\n        file_link_text = file_link_text.replace(\".md\", \"\").replace(/#/g, \" > \");\n        // create item\n        const item = list.createEl(\"div\", { cls: search_result_class });\n        // create span for toggle\n        const toggle = item.createEl(\"span\", { cls: \"is-clickable\" });\n        // insert right triangle svg as toggle\n        Obsidian.setIcon(toggle, \"right-triangle\"); // must come before adding other elms to prevent overwrite\n        const link = toggle.createEl(\"a\", {\n          cls: \"search-result-file-title\",\n          title: nearest[i].link,\n        });\n        link.innerHTML = file_link_text;\n        // add listeners to link\n        this.add_link_listeners(link, nearest[i], item);\n        toggle.addEventListener(\"click\", (event) => {\n          // find parent containing search-result class\n          let parent = event.target.parentElement;\n          while (!parent.classList.contains(\"search-result\")) {\n            parent = parent.parentElement;\n          }\n          // toggle sc-collapsed class\n          parent.classList.toggle(\"sc-collapsed\");\n        });\n        const contents = item.createEl(\"ul\", { cls: \"\" });\n        const contents_container = contents.createEl(\"li\", {\n          cls: \"search-result-file-title is-clickable\",\n          title: nearest[i].link,\n        });\n        if(nearest[i].link.indexOf(\"#\") > -1){ // is block\n          Obsidian.MarkdownRenderer.renderMarkdown((await this.block_retriever(nearest[i].link, {lines: 10, max_chars: 1000})), contents_container, nearest[i].link, new Obsidian.Component());\n        }else{ // is file\n          const first_ten_lines = await this.file_retriever(nearest[i].link, {lines: 10, max_chars: 1000});\n          if(!first_ten_lines) continue; // skip if file is empty\n          Obsidian.MarkdownRenderer.renderMarkdown(first_ten_lines, contents_container, nearest[i].link, new Obsidian.Component());\n        }\n        this.add_link_listeners(contents, nearest[i], item);\n      }\n      this.render_brand(container, \"block\");\n      return;\n    }\n\n    // group nearest by file\n    const nearest_by_file = {};\n    for (let i = 0; i < nearest.length; i++) {\n      const curr = nearest[i];\n      const link = curr.link;\n      // skip if link is an object (indicates external logic)\n      if (typeof link === \"object\") {\n        nearest_by_file[link.path] = [curr];\n        continue;\n      }\n      if (link.indexOf(\"#\") > -1) {\n        const file_path = link.split(\"#\")[0];\n        if (!nearest_by_file[file_path]) {\n          nearest_by_file[file_path] = [];\n        }\n        nearest_by_file[file_path].push(nearest[i]);\n      } else {\n        if (!nearest_by_file[link]) {\n          nearest_by_file[link] = [];\n        }\n        // always add to front of array\n        nearest_by_file[link].unshift(nearest[i]);\n      }\n    }\n    // for each file\n    const keys = Object.keys(nearest_by_file);\n    for (let i = 0; i < keys.length; i++) {\n      const file = nearest_by_file[keys[i]];\n      /**\n       * Begin external link handling\n       */\n      // if link is an object (indicates v2 logic)\n      if (typeof file[0].link === \"object\") {\n        const curr = file[0];\n        const meta = curr.link;\n        if (meta.path.startsWith(\"http\")) {\n          const item = list.createEl(\"div\", { cls: \"search-result\" });\n          const link = item.createEl(\"a\", {\n            cls: \"search-result-file-title is-clickable\",\n            href: meta.path,\n            title: meta.title,\n          });\n          link.innerHTML = this.render_external_link_elm(meta);\n          item.setAttr('draggable', 'true');\n          continue; // ends here for external links\n        }\n      }\n      /**\n       * Handles Internal\n       */\n      let file_link_text;\n      const file_similarity_pct = Math.round(file[0].similarity * 100) + \"%\";\n      if (this.settings.show_full_path) {\n        const pcs = file[0].link.split(\"/\");\n        file_link_text = pcs[pcs.length - 1];\n        const path = pcs.slice(0, pcs.length - 1).join(\"/\");\n        file_link_text = `<small>${path} | ${file_similarity_pct}</small><br>${file_link_text}`;\n      } else {\n        file_link_text = file[0].link.split(\"/\").pop();\n        // add similarity percentage\n        file_link_text += ' | ' + file_similarity_pct;\n      }\n\n\n        \n      // skip contents rendering if incompatible file type\n      // ex. not markdown or contains '.excalidraw'\n      if(!this.renderable_file_type(file[0].link)) {\n        const item = list.createEl(\"div\", { cls: \"search-result\" });\n        const file_link = item.createEl(\"a\", {\n          cls: \"search-result-file-title is-clickable\",\n          title: file[0].link,\n        });\n        file_link.innerHTML = file_link_text;\n        // add link listeners to file link\n        this.add_link_listeners(file_link, file[0], item);\n        continue;\n      }\n\n\n      // remove file extension if .md\n      file_link_text = file_link_text.replace(\".md\", \"\").replace(/#/g, \" > \");\n      const item = list.createEl(\"div\", { cls: search_result_class });\n      const toggle = item.createEl(\"span\", { cls: \"is-clickable\" });\n      // insert right triangle svg icon as toggle button in span\n      Obsidian.setIcon(toggle, \"right-triangle\"); // must come before adding other elms else overwrites\n      const file_link = toggle.createEl(\"a\", {\n        cls: \"search-result-file-title\",\n        title: file[0].link,\n      });\n      file_link.innerHTML = file_link_text;\n      // add link listeners to file link\n      this.add_link_listeners(file_link, file[0], toggle);\n      toggle.addEventListener(\"click\", (event) => {\n        // find parent containing class search-result\n        let parent = event.target;\n        while (!parent.classList.contains(\"search-result\")) {\n          parent = parent.parentElement;\n        }\n        parent.classList.toggle(\"sc-collapsed\");\n        // TODO: if block container is empty, render markdown from block retriever\n      });\n      const file_link_list = item.createEl(\"ul\");\n      // for each link in file\n      for (let j = 0; j < file.length; j++) {\n        // if is a block (has # in link)\n        if(file[j].link.indexOf(\"#\") > -1) {\n          const block = file[j];\n          const block_link = file_link_list.createEl(\"li\", {\n            cls: \"search-result-file-title is-clickable\",\n            title: block.link,\n          });\n          // skip block context if file.length === 1 because already added\n          if(file.length > 1) {\n            const block_context = this.render_block_context(block);\n            const block_similarity_pct = Math.round(block.similarity * 100) + \"%\";\n            block_link.innerHTML = `<small>${block_context} | ${block_similarity_pct}</small>`;\n          }\n          const block_container = block_link.createEl(\"div\");\n          // TODO: move to rendering on expanding section (toggle collapsed)\n          Obsidian.MarkdownRenderer.renderMarkdown((await this.block_retriever(block.link, {lines: 10, max_chars: 1000})), block_container, block.link, new Obsidian.Component());\n          // add link listeners to block link\n          this.add_link_listeners(block_link, block, file_link_list);\n        }else{\n          // get first ten lines of file\n          const file_link_list = item.createEl(\"ul\");\n          const block_link = file_link_list.createEl(\"li\", {\n            cls: \"search-result-file-title is-clickable\",\n            title: file[0].link,\n          });\n          const block_container = block_link.createEl(\"div\");\n          let first_ten_lines = await this.file_retriever(file[0].link, {lines: 10, max_chars: 1000});\n          if(!first_ten_lines) continue; // if file not found, skip\n          Obsidian.MarkdownRenderer.renderMarkdown(first_ten_lines, block_container, file[0].link, new Obsidian.Component());\n          this.add_link_listeners(block_link, file[0], file_link_list);\n\n        }\n      }\n    }\n    this.render_brand(container, \"file\");\n  }\n\n  add_link_listeners(item, curr, list) {\n    item.addEventListener(\"click\", async (event) => {\n      await this.open_note(curr, event);\n    });\n    // drag-on\n    // currently only works with full-file links\n    item.setAttr('draggable', 'true');\n    item.addEventListener('dragstart', (event) => {\n      const dragManager = this.app.dragManager;\n      const file_path = curr.link.split(\"#\")[0];\n      const file = this.app.metadataCache.getFirstLinkpathDest(file_path, '');\n      const dragData = dragManager.dragFile(event, file);\n      // console.log(dragData);\n      dragManager.onDragStart(event, dragData);\n    });\n    // if curr.link contains curly braces, return (incompatible with hover-link)\n    if (curr.link.indexOf(\"{\") > -1) return;\n    // trigger hover event on link\n    item.addEventListener(\"mouseover\", (event) => {\n      this.app.workspace.trigger(\"hover-link\", {\n        event,\n        source: SMART_CONNECTIONS_VIEW_TYPE,\n        hoverParent: list,\n        targetEl: item,\n        linktext: curr.link,\n      });\n    });\n  }\n\n  // get target file from link path\n  // if sub-section is linked, open file and scroll to sub-section\n  async open_note(curr, event=null) {\n    let targetFile;\n    let heading;\n    if (curr.link.indexOf(\"#\") > -1) {\n      // remove after # from link\n      targetFile = this.app.metadataCache.getFirstLinkpathDest(curr.link.split(\"#\")[0], \"\");\n      // console.log(targetFile);\n      const target_file_cache = this.app.metadataCache.getFileCache(targetFile);\n      // console.log(target_file_cache);\n      // get heading\n      let heading_text = curr.link.split(\"#\").pop();\n      // if heading text contains a curly brace, get the number inside the curly braces as occurence\n      let occurence = 0;\n      if (heading_text.indexOf(\"{\") > -1) {\n        // get occurence\n        occurence = parseInt(heading_text.split(\"{\")[1].split(\"}\")[0]);\n        // remove occurence from heading text\n        heading_text = heading_text.split(\"{\")[0];\n      }\n      // get headings from file cache\n      const headings = target_file_cache.headings;\n      // get headings with the same depth and text as the link\n      for(let i = 0; i < headings.length; i++) {\n        if (headings[i].heading === heading_text) {\n          // if occurence is 0, set heading and break\n          if(occurence === 0) {\n            heading = headings[i];\n            break;\n          }\n          occurence--; // decrement occurence\n        }\n      }\n      // console.log(heading);\n    } else {\n      targetFile = this.app.metadataCache.getFirstLinkpathDest(curr.link, \"\");\n    }\n    let leaf;\n    if(event) {\n      // properly handle if the meta/ctrl key is pressed\n      const mod = Obsidian.Keymap.isModEvent(event);\n      // get most recent leaf\n      leaf = this.app.workspace.getLeaf(mod);\n    }else{\n      // get most recent leaf\n      leaf = this.app.workspace.getMostRecentLeaf();\n    }\n    await leaf.openFile(targetFile);\n    if (heading) {\n      let { editor } = leaf.view;\n      const pos = { line: heading.position.start.line, ch: 0 };\n      editor.setCursor(pos);\n      editor.scrollIntoView({ to: pos, from: pos }, true);\n    }\n  }\n\n  render_block_context(block) {\n    const block_headings = block.link.split(\".md\")[1].split(\"#\");\n    // starting with the last heading first, iterate through headings\n    let block_context = \"\";\n    for (let i = block_headings.length - 1; i >= 0; i--) {\n      if(block_context.length > 0) {\n        block_context = ` > ${block_context}`;\n      }\n      block_context = block_headings[i] + block_context;\n      // if block context is longer than N characters, break\n      if (block_context.length > 100) {\n        break;\n      }\n    }\n    // remove leading > if exists\n    if (block_context.startsWith(\" > \")) {\n      block_context = block_context.slice(3);\n    }\n    return block_context;\n\n  }\n\n  renderable_file_type(link) {\n    return (link.indexOf(\".md\") !== -1) && (link.indexOf(\".excalidraw\") === -1);\n  }\n\n  render_external_link_elm(meta){\n    if(meta.source) {\n      if(meta.source === \"Gmail\") meta.source = \"\uD83D\uDCE7 Gmail\";\n      return `<small>${meta.source}</small><br>${meta.title}`;\n    }\n    // remove http(s)://\n    let domain = meta.path.replace(/(^\\w+:|^)\\/\\//, \"\");\n    // separate domain from path\n    domain = domain.split(\"/\")[0];\n    // wrap domain in <small> and add line break\n    return `<small>\uD83C\uDF10 ${domain}</small><br>${meta.title}`;\n  }\n  // get all folders\n  async get_all_folders() {\n    if(!this.folders || this.folders.length === 0){\n      this.folders = await this.get_folders();\n    }\n    return this.folders;\n  }\n  // get folders, traverse non-hidden sub-folders\n  async get_folders(path = \"/\") {\n    let folders = (await this.app.vault.adapter.list(path)).folders;\n    let folder_list = [];\n    for (let i = 0; i < folders.length; i++) {\n      if (folders[i].startsWith(\".\")) continue;\n      folder_list.push(folders[i]);\n      folder_list = folder_list.concat(await this.get_folders(folders[i] + \"/\"));\n    }\n    return folder_list;\n  }\n\n\n  async sync_notes() {\n    // if license key is not set, return\n    if(!this.settings.license_key){\n      new Obsidian.Notice(\"Smart Connections: Supporter license key is required to sync notes to the ChatGPT Plugin server.\");\n      return;\n    }\n    console.log(\"syncing notes\");\n    // get all files in vault\n    const files = this.app.vault.getMarkdownFiles().filter((file) => {\n      // filter out file paths matching any strings in this.file_exclusions\n      for(let i = 0; i < this.file_exclusions.length; i++) {\n        if(file.path.indexOf(this.file_exclusions[i]) > -1) {\n          return false;\n        }\n      }\n      return true;\n    });\n    const notes = await this.build_notes_object(files);\n    console.log(\"object built\");\n    // save notes object to .smart-connections/notes.json\n    await this.app.vault.adapter.write(\".smart-connections/notes.json\", JSON.stringify(notes, null, 2));\n    console.log(\"notes saved\");\n    console.log(this.settings.license_key);\n    // POST notes object to server\n    const response = await (0, Obsidian.requestUrl)({\n      url: \"https://sync.smartconnections.app/sync\",\n      method: \"POST\",\n      headers: {\n        \"Content-Type\": \"application/json\",\n      },\n      contentType: \"application/json\",\n      body: JSON.stringify({\n        license_key: this.settings.license_key,\n        notes: notes\n      })\n    });\n    console.log(response);\n\n  }\n\n  async build_notes_object(files) {\n    let output = {};\n  \n    for(let i = 0; i < files.length; i++) {\n      let file = files[i];\n      let parts = file.path.split(\"/\");\n      let current = output;\n  \n      for (let ii = 0; ii < parts.length; ii++) {\n        let part = parts[ii];\n  \n        if (ii === parts.length - 1) {\n          // This is a file\n          current[part] = await this.app.vault.cachedRead(file);\n        } else {\n          // This is a directory\n          if (!current[part]) {\n            current[part] = {};\n          }\n  \n          current = current[part];\n        }\n      }\n    }\n  \n    return output;\n  }\n\n}\n\nconst SMART_CONNECTIONS_VIEW_TYPE = \"smart-connections-view\";\nclass SmartConnectionsView extends Obsidian.ItemView {\n  constructor(leaf, plugin) {\n    super(leaf);\n    this.plugin = plugin;\n    this.nearest = null;\n    this.load_wait = null;\n  }\n  getViewType() {\n    return SMART_CONNECTIONS_VIEW_TYPE;\n  }\n\n  getDisplayText() {\n    return \"Smart Connections Files\";\n  }\n\n  getIcon() {\n    return \"smart-connections\";\n  }\n\n\n  set_message(message) {\n    const container = this.containerEl.children[1];\n    // clear container\n    container.empty();\n    // initiate top bar\n    this.initiate_top_bar(container);\n    // if mesage is an array, loop through and create a new p element for each message\n    if (Array.isArray(message)) {\n      for (let i = 0; i < message.length; i++) {\n        container.createEl(\"p\", { cls: \"sc_message\", text: message[i] });\n      }\n    }else{\n      // create p element with message\n      container.createEl(\"p\", { cls: \"sc_message\", text: message });\n    }\n  }\n  render_link_text(link, show_full_path=false) {\n    /**\n     * Begin internal links\n     */\n    // if show full path is false, remove file path\n    if (!show_full_path) {\n      link = link.split(\"/\").pop();\n    }\n    // if contains '#'\n    if (link.indexOf(\"#\") > -1) {\n      // split at .md\n      link = link.split(\".md\");\n      // wrap first part in <small> and add line break\n      link[0] = `<small>${link[0]}</small><br>`;\n      // join back together\n      link = link.join(\"\");\n      // replace '#' with ' \u00BB '\n      link = link.replace(/\\#/g, \" \u00BB \");\n    }else{\n      // remove '.md'\n      link = link.replace(\".md\", \"\");\n    }\n    return link;\n  }\n\n\n  set_nearest(nearest, nearest_context=null, results_only=false) {\n    // get container element\n    const container = this.containerEl.children[1];\n    // if results only is false, clear container and initiate top bar\n    if(!results_only){\n      // clear container\n      container.empty();\n      this.initiate_top_bar(container, nearest_context);\n    }\n    // update results\n    this.plugin.update_results(container, nearest);\n  }\n\n  initiate_top_bar(container, nearest_context=null) {\n    let top_bar;\n    // if top bar already exists, empty it\n    if ((container.children.length > 0) && (container.children[0].classList.contains(\"sc-top-bar\"))) {\n      top_bar = container.children[0];\n      top_bar.empty();\n    } else {\n      // init container for top bar\n      top_bar = container.createEl(\"div\", { cls: \"sc-top-bar\" });\n    }\n    // if highlighted text is not null, create p element with highlighted text\n    if (nearest_context) {\n      top_bar.createEl(\"p\", { cls: \"sc-context\", text: nearest_context });\n    }\n    // add chat button\n    const chat_button = top_bar.createEl(\"button\", { cls: \"sc-chat-button\" });\n    // add icon to chat button\n    Obsidian.setIcon(chat_button, \"message-square\");\n    // add click listener to chat button\n    chat_button.addEventListener(\"click\", () => {\n      // open chat\n      this.plugin.open_chat();\n    });\n    // add search button\n    const search_button = top_bar.createEl(\"button\", { cls: \"sc-search-button\" });\n    // add icon to search button\n    Obsidian.setIcon(search_button, \"search\");\n    // add click listener to search button\n    search_button.addEventListener(\"click\", () => {\n      // empty top bar\n      top_bar.empty();\n      // create input element\n      const search_container = top_bar.createEl(\"div\", { cls: \"search-input-container\" });\n      const input = search_container.createEl(\"input\", {\n        cls: \"sc-search-input\",\n        type: \"search\",\n        placeholder: \"Type to start search...\", \n      });\n      // focus input\n      input.focus();\n      // add keydown listener to input\n      input.addEventListener(\"keydown\", (event) => {\n        // if escape key is pressed\n        if (event.key === \"Escape\") {\n          this.clear_auto_searcher();\n          // clear top bar\n          this.initiate_top_bar(container, nearest_context);\n        }\n      });\n\n      // add keyup listener to input\n      input.addEventListener(\"keyup\", (event) => {\n        // if this.search_timeout is not null then clear it and set to null\n        this.clear_auto_searcher();\n        // get search term\n        const search_term = input.value;\n        // if enter key is pressed\n        if (event.key === \"Enter\" && search_term !== \"\") {\n          this.search(search_term);\n        }\n        // if any other key is pressed and input is not empty then wait 500ms and make_connections\n        else if (search_term !== \"\") {\n          // clear timeout\n          clearTimeout(this.search_timeout);\n          // set timeout\n          this.search_timeout = setTimeout(() => {\n            this.search(search_term, true);\n          }, 700);\n        }\n      });\n    });\n  }\n\n  // render buttons: \"create\" and \"retry\" for loading embeddings.json file\n  render_embeddings_buttons() {\n    // get container element\n    const container = this.containerEl.children[1];\n    // clear container\n    container.empty();\n    // create heading that says \"Embeddings file not found\"\n    container.createEl(\"h2\", { cls: \"scHeading\", text: \"Embeddings file not found\" });\n    // create div for buttons\n    const button_div = container.createEl(\"div\", { cls: \"scButtonDiv\" });\n    // create \"create\" button\n    const create_button = button_div.createEl(\"button\", { cls: \"scButton\", text: \"Create embeddings.json\" });\n    // note that creating embeddings.json file will trigger bulk embedding and may take a while\n    button_div.createEl(\"p\", { cls: \"scButtonNote\", text: \"Warning: Creating embeddings.json file will trigger bulk embedding and may take a while\" });\n    // create \"retry\" button\n    const retry_button = button_div.createEl(\"button\", { cls: \"scButton\", text: \"Retry\" });\n    // try to load embeddings.json file again\n    button_div.createEl(\"p\", { cls: \"scButtonNote\", text: \"If embeddings.json file already exists, click 'Retry' to load it\" });\n\n    // add click event to \"create\" button\n    create_button.addEventListener(\"click\", async (event) => {\n      // create embeddings.json file\n      await this.plugin.smart_vec_lite.init_embeddings_file();\n      // reload view\n      await this.render_connections();\n    });\n\n    // add click event to \"retry\" button\n    retry_button.addEventListener(\"click\", async (event) => {\n      console.log(\"retrying to load embeddings.json file\");\n      // reload embeddings.json file\n      await this.plugin.init_vecs();\n      // reload view\n      await this.render_connections();\n    });\n  }\n\n  async onOpen() {\n    const container = this.containerEl.children[1];\n    container.empty();\n    // placeholder text\n    container.createEl(\"p\", { cls: \"scPlaceholder\", text: \"Open a note to find connections.\" }); \n\n    // runs when file is opened\n    this.plugin.registerEvent(this.app.workspace.on('file-open', (file) => {\n      // if no file is open, return\n      if(!file) {\n        // console.log(\"no file open, returning\");\n        return;\n      }\n      // return if file type is not supported\n      if(SUPPORTED_FILE_TYPES.indexOf(file.extension) === -1) {\n        return this.set_message([\n          \"File: \"+file.name\n          ,\"Unsupported file type (Supported: \"+SUPPORTED_FILE_TYPES.join(\", \")+\")\"\n        ]);\n      }\n      // run render_connections after 1 second to allow for file to load\n      if(this.load_wait){\n        clearTimeout(this.load_wait);\n      }\n      this.load_wait = setTimeout(() => {\n        this.render_connections(file);\n        this.load_wait = null;\n      }, 1000);\n        \n    }));\n\n    this.app.workspace.registerHoverLinkSource(SMART_CONNECTIONS_VIEW_TYPE, {\n        display: 'Smart Connections Files',\n        defaultMod: true,\n    });\n    this.app.workspace.registerHoverLinkSource(SMART_CONNECTIONS_CHAT_VIEW_TYPE, {\n        display: 'Smart Chat Links',\n        defaultMod: true,\n    });\n\n    this.app.workspace.onLayoutReady(this.initialize.bind(this));\n    \n  }\n  \n  async initialize() {\n    this.set_message(\"Loading embeddings file...\");\n    const vecs_intiated = await this.plugin.init_vecs();\n    if(vecs_intiated){\n      this.set_message(\"Embeddings file loaded.\");\n      await this.render_connections();\n    }else{\n      this.render_embeddings_buttons();\n    }\n\n    /**\n     * EXPERIMENTAL\n     * - window-based API access\n     * - code-block rendering\n     */\n    this.api = new SmartConnectionsViewApi(this.app, this.plugin, this);\n    // register API to global window object\n    (window[\"SmartConnectionsViewApi\"] = this.api) && this.register(() => delete window[\"SmartConnectionsViewApi\"]);\n\n  }\n\n  async onClose() {\n    console.log(\"closing smart connections view\");\n    this.app.workspace.unregisterHoverLinkSource(SMART_CONNECTIONS_VIEW_TYPE);\n    this.plugin.view = null;\n  }\n\n  async render_connections(context=null) {\n    console.log(\"rendering connections\");\n    // if API key is not set then update view message\n    if(!this.plugin.settings.api_key) {\n      this.set_message(\"An OpenAI API key is required to make Smart Connections\");\n      return;\n    }\n    if(!this.plugin.embeddings_loaded){\n      await this.plugin.init_vecs();\n    }\n    // if embedding still not loaded, return\n    if(!this.plugin.embeddings_loaded) {\n      console.log(\"embeddings files still not loaded or yet to be created\");\n      this.render_embeddings_buttons();\n      return;\n    }\n    this.set_message(\"Making Smart Connections...\");\n    /**\n     * Begin highlighted-text-level search\n     */\n    if(typeof context === \"string\") {\n      const highlighted_text = context;\n      // get embedding for highlighted text\n      await this.search(highlighted_text);\n      return; // ends here if context is a string\n    }\n\n    /** \n     * Begin file-level search\n     */    \n    this.nearest = null;\n    this.interval_count = 0;\n    this.rendering = false;\n    this.file = context;\n    // if this.interval is set then clear it\n    if(this.interval) {\n      clearInterval(this.interval);\n      this.interval = null;\n    }\n    // set interval to check if nearest is set\n    this.interval = setInterval(() => {\n      if(!this.rendering){\n        if(this.file instanceof Obsidian.TFile) {\n          this.rendering = true;\n          this.render_note_connections(this.file);\n        }else{\n          // get current note\n          this.file = this.app.workspace.getActiveFile();\n          // if still no current note then return\n          if(!this.file && this.count > 1) {\n            clearInterval(this.interval);\n            this.set_message(\"No active file\");\n            return; \n          }\n        }\n      }else{\n        if(this.nearest) {\n          clearInterval(this.interval);\n          // if nearest is a string then update view message\n          if (typeof this.nearest === \"string\") {\n            this.set_message(this.nearest);\n          } else {\n            // set nearest connections\n            this.set_nearest(this.nearest, \"File: \" + this.file.name);\n          }\n          // if render_log.failed_embeddings then update failed_embeddings.txt\n          if (this.plugin.render_log.failed_embeddings.length > 0) {\n            this.plugin.save_failed_embeddings();\n          }\n          // get object keys of render_log\n          this.plugin.output_render_log();\n          return; \n        }else{\n          this.interval_count++;\n          this.set_message(\"Making Smart Connections...\"+this.interval_count);\n        }\n      }\n    }, 10);\n  }\n\n  async render_note_connections(file) {\n    this.nearest = await this.plugin.find_note_connections(file);\n  }\n\n  clear_auto_searcher() {\n    if (this.search_timeout) {\n      clearTimeout(this.search_timeout);\n      this.search_timeout = null;\n    }\n  }\n\n  async search(search_text, results_only=false) {\n    const nearest = await this.plugin.api.search(search_text);\n    // render results in view with first 100 characters of search text\n    const nearest_context = `Selection: \"${search_text.length > 100 ? search_text.substring(0, 100) + \"...\" : search_text}\"`;\n    this.set_nearest(nearest, nearest_context, results_only);\n  }\n\n}\nclass SmartConnectionsViewApi {\n  constructor(app, plugin, view) {\n    this.app = app;\n    this.plugin = plugin;\n    this.view = view;\n  }\n  async search (search_text) {\n    return await this.plugin.api.search(search_text);\n  }\n  // trigger reload of embeddings file\n  async reload_embeddings_file() {\n    await this.plugin.init_vecs();\n    await this.view.render_connections();\n  }\n}\nclass ScSearchApi {\n  constructor(app, plugin) {\n    this.app = app;\n    this.plugin = plugin;\n  }\n  async search (search_text, filter={}) {\n    filter = {\n      skip_sections: this.plugin.settings.skip_sections,\n      ...filter\n    }\n    let nearest = [];\n    const resp = await this.plugin.request_embedding_from_input(search_text);\n    if (resp && resp.data && resp.data[0] && resp.data[0].embedding) {\n      nearest = this.plugin.smart_vec_lite.nearest(resp.data[0].embedding, filter);\n    } else {\n      // resp is null, undefined, or missing data\n      new Obsidian.Notice(\"Smart Connections: Error getting embedding\");\n    }\n    return nearest;\n  }\n}\n\nclass SmartConnectionsSettingsTab extends Obsidian.PluginSettingTab {\n  constructor(app, plugin) {\n    super(app, plugin);\n    this.plugin = plugin;\n  }\n  display() {\n    const {\n      containerEl\n    } = this;\n    containerEl.empty();\n    containerEl.createEl(\"h2\", {\n      text: \"Supporter Settings\"\n    });\n    // list supporter benefits\n    containerEl.createEl(\"p\", {\n      text: \"As a Smart Connections \\\"Supporter\\\", fast-track your PKM journey with priority perks and pioneering innovations.\"\n    });\n    // three list items\n    const supporter_benefits_list = containerEl.createEl(\"ul\");\n    supporter_benefits_list.createEl(\"li\", {\n      text: \"Enjoy swift, top-priority support.\"\n    });\n    supporter_benefits_list.createEl(\"li\", {\n      text: \"Gain early access to version 2 (includes local embedding model).\"\n    });\n    supporter_benefits_list.createEl(\"li\", {\n      text: \"Stay informed and engaged with exclusive supporter-only communications.\"\n    });\n    // add a text input to enter supporter license key\n    new Obsidian.Setting(containerEl).setName(\"Supporter License Key\").setDesc(\"Note: this is not required to use Smart Connections.\").addText((text) => text.setPlaceholder(\"Enter your license_key\").setValue(this.plugin.settings.license_key).onChange(async (value) => {\n      this.plugin.settings.license_key = value.trim();\n      await this.plugin.saveSettings(true);\n    }));\n    // button \"get v2\"\n    new Obsidian.Setting(containerEl).setName(\"Get v2\").setDesc(\"Get v2 (warning: very early beta release, likely to crash, please send issues directly to the supporter email for quick response)\").addButton((button) => button.setButtonText(\"Get v2 (unstable)\").onClick(async () => {\n      await this.plugin.update_to_v2();\n    }));\n    // add button to trigger sync notes to use with ChatGPT\n    new Obsidian.Setting(containerEl).setName(\"Sync Notes\").setDesc(\"Make notes available via the Smart Connections ChatGPT Plugin. Respects exclusion settings configured below.\").addButton((button) => button.setButtonText(\"Sync Notes\").onClick(async () => {\n      // sync notes\n      await this.plugin.sync_notes();\n    }));\n    // add button to become a supporter\n    new Obsidian.Setting(containerEl).setName(\"Become a Supporter\").setDesc(\"Become a Supporter\").addButton((button) => button.setButtonText(\"Become a Supporter\").onClick(async () => {\n      const payment_pages = [\n        \"https://buy.stripe.com/9AQ5kO5QnbAWgGAbIY\",\n        \"https://buy.stripe.com/9AQ7sWemT48u1LGcN4\"\n      ];\n      if(!this.plugin.payment_page_index){\n        this.plugin.payment_page_index = Math.round(Math.random());\n      }\n      // open supporter page in browser\n      window.open(payment_pages[this.plugin.payment_page_index]);\n    }));\n\n    \n    containerEl.createEl(\"h2\", {\n      text: \"OpenAI Settings\"\n    });\n    // add a text input to enter the API key\n    new Obsidian.Setting(containerEl).setName(\"OpenAI API Key\").setDesc(\"Required: an OpenAI API key is currently required to use Smart Connections.\").addText((text) => text.setPlaceholder(\"Enter your api_key\").setValue(this.plugin.settings.api_key).onChange(async (value) => {\n      this.plugin.settings.api_key = value.trim();\n      await this.plugin.saveSettings(true);\n    }));\n    // add a button to test the API key is working\n    new Obsidian.Setting(containerEl).setName(\"Test API Key\").setDesc(\"Test API Key\").addButton((button) => button.setButtonText(\"Test API Key\").onClick(async () => {\n      // test API key\n      const resp = await this.plugin.test_api_key();\n      if(resp) {\n        new Obsidian.Notice(\"Smart Connections: API key is valid\");\n      }else{\n        new Obsidian.Notice(\"Smart Connections: API key is not working as expected!\");\n      }\n    }));\n    // add dropdown to select the model\n    new Obsidian.Setting(containerEl).setName(\"Smart Chat Model\").setDesc(\"Select a model to use with Smart Chat.\").addDropdown((dropdown) => {\n      dropdown.addOption(\"gpt-3.5-turbo-16k\", \"gpt-3.5-turbo-16k\");\n      dropdown.addOption(\"gpt-4\", \"gpt-4 (limited access, 8k)\");\n      dropdown.addOption(\"gpt-3.5-turbo\", \"gpt-3.5-turbo (4k)\");\n      dropdown.addOption(\"gpt-4-1106-preview\", \"gpt-4-turbo (128k)\");\n      dropdown.onChange(async (value) => {\n        this.plugin.settings.smart_chat_model = value;\n        await this.plugin.saveSettings();\n      });\n      dropdown.setValue(this.plugin.settings.smart_chat_model);\n    });\n    // language\n    new Obsidian.Setting(containerEl).setName(\"Default Language\").setDesc(\"Default language to use for Smart Chat. Changes which self-referential pronouns will trigger lookup of your notes.\").addDropdown((dropdown) => {\n      // get Object keys from pronous\n      const languages = Object.keys(SMART_TRANSLATION);\n      for(let i = 0; i < languages.length; i++) {\n        dropdown.addOption(languages[i], languages[i]);\n      }\n      dropdown.onChange(async (value) => {\n        this.plugin.settings.language = value;\n        await this.plugin.saveSettings();\n        self_ref_pronouns_list.setText(this.get_self_ref_list());\n        // if chat view is open then run new_chat()\n        const chat_view = this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE).length > 0 ? this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE)[0].view : null;\n        if(chat_view) {\n          chat_view.new_chat();\n        }\n      });\n      dropdown.setValue(this.plugin.settings.language);\n    });\n    // list current self-referential pronouns\n    const self_ref_pronouns_list = containerEl.createEl(\"span\", {\n      text: this.get_self_ref_list()\n    });\n    containerEl.createEl(\"h2\", {\n      text: \"Exclusions\"\n    });\n    // list file exclusions\n    new Obsidian.Setting(containerEl).setName(\"file_exclusions\").setDesc(\"'Excluded file' matchers separated by a comma.\").addText((text) => text.setPlaceholder(\"drawings,prompts/logs\").setValue(this.plugin.settings.file_exclusions).onChange(async (value) => {\n      this.plugin.settings.file_exclusions = value;\n      await this.plugin.saveSettings();\n    }));\n    // list folder exclusions\n    new Obsidian.Setting(containerEl).setName(\"folder_exclusions\").setDesc(\"'Excluded folder' matchers separated by a comma.\").addText((text) => text.setPlaceholder(\"drawings,prompts/logs\").setValue(this.plugin.settings.folder_exclusions).onChange(async (value) => {\n      this.plugin.settings.folder_exclusions = value;\n      await this.plugin.saveSettings();\n    }));\n    // list path only matchers\n    new Obsidian.Setting(containerEl).setName(\"path_only\").setDesc(\"'Path only' matchers separated by a comma.\").addText((text) => text.setPlaceholder(\"drawings,prompts/logs\").setValue(this.plugin.settings.path_only).onChange(async (value) => {\n      this.plugin.settings.path_only = value;\n      await this.plugin.saveSettings();\n    }));\n    // list header exclusions\n    new Obsidian.Setting(containerEl).setName(\"header_exclusions\").setDesc(\"'Excluded header' matchers separated by a comma. Works for 'blocks' only.\").addText((text) => text.setPlaceholder(\"drawings,prompts/logs\").setValue(this.plugin.settings.header_exclusions).onChange(async (value) => {\n      this.plugin.settings.header_exclusions = value;\n      await this.plugin.saveSettings();\n    }));\n    containerEl.createEl(\"h2\", {\n      text: \"Display\"\n    });\n    // toggle showing full path in view\n    new Obsidian.Setting(containerEl).setName(\"show_full_path\").setDesc(\"Show full path in view.\").addToggle((toggle) => toggle.setValue(this.plugin.settings.show_full_path).onChange(async (value) => {\n      this.plugin.settings.show_full_path = value;\n      await this.plugin.saveSettings(true);\n    }));\n    // toggle expanded view by default\n    new Obsidian.Setting(containerEl).setName(\"expanded_view\").setDesc(\"Expanded view by default.\").addToggle((toggle) => toggle.setValue(this.plugin.settings.expanded_view).onChange(async (value) => {\n      this.plugin.settings.expanded_view = value;\n      await this.plugin.saveSettings(true);\n    }));\n    // toggle group nearest by file\n    new Obsidian.Setting(containerEl).setName(\"group_nearest_by_file\").setDesc(\"Group nearest by file.\").addToggle((toggle) => toggle.setValue(this.plugin.settings.group_nearest_by_file).onChange(async (value) => {\n      this.plugin.settings.group_nearest_by_file = value;\n      await this.plugin.saveSettings(true);\n    }));\n    // toggle view_open on Obsidian startup\n    new Obsidian.Setting(containerEl).setName(\"view_open\").setDesc(\"Open view on Obsidian startup.\").addToggle((toggle) => toggle.setValue(this.plugin.settings.view_open).onChange(async (value) => {\n      this.plugin.settings.view_open = value;\n      await this.plugin.saveSettings(true);\n    }));\n    // toggle chat_open on Obsidian startup\n    new Obsidian.Setting(containerEl).setName(\"chat_open\").setDesc(\"Open view on Obsidian startup.\").addToggle((toggle) => toggle.setValue(this.plugin.settings.chat_open).onChange(async (value) => {\n      this.plugin.settings.chat_open = value;\n      await this.plugin.saveSettings(true);\n    }));\n    containerEl.createEl(\"h2\", {\n      text: \"Advanced\"\n    });\n    // toggle log_render\n    new Obsidian.Setting(containerEl).setName(\"log_render\").setDesc(\"Log render details to console (includes token_usage).\").addToggle((toggle) => toggle.setValue(this.plugin.settings.log_render).onChange(async (value) => {\n      this.plugin.settings.log_render = value;\n      await this.plugin.saveSettings(true);\n    }));\n    // toggle files in log_render\n    new Obsidian.Setting(containerEl).setName(\"log_render_files\").setDesc(\"Log embedded objects paths with log render (for debugging).\").addToggle((toggle) => toggle.setValue(this.plugin.settings.log_render_files).onChange(async (value) => {\n      this.plugin.settings.log_render_files = value;\n      await this.plugin.saveSettings(true);\n    }));\n    // toggle skip_sections\n    new Obsidian.Setting(containerEl).setName(\"skip_sections\").setDesc(\"Skips making connections to specific sections within notes. Warning: reduces usefulness for large files and requires 'Force Refresh' for sections to work in the future.\").addToggle((toggle) => toggle.setValue(this.plugin.settings.skip_sections).onChange(async (value) => {\n      this.plugin.settings.skip_sections = value;\n      await this.plugin.saveSettings(true);\n    }));\n    // test file writing by creating a test file, then writing additional data to the file, and returning any error text if it fails\n    containerEl.createEl(\"h3\", {\n      text: \"Test File Writing\"\n    });\n    // manual save button\n    containerEl.createEl(\"h3\", {\n      text: \"Manual Save\"\n    });\n    let manual_save_results = containerEl.createEl(\"div\");\n    new Obsidian.Setting(containerEl).setName(\"manual_save\").setDesc(\"Save current embeddings\").addButton((button) => button.setButtonText(\"Manual Save\").onClick(async () => {\n      // confirm\n      if (confirm(\"Are you sure you want to save your current embeddings?\")) {\n        // save\n        try{\n          await this.plugin.save_embeddings_to_file(true);\n          manual_save_results.innerHTML = \"Embeddings saved successfully.\";\n        }catch(e){\n          manual_save_results.innerHTML = \"Embeddings failed to save. Error: \" + e;\n        }\n      }\n    }));\n\n    // list previously failed files\n    containerEl.createEl(\"h3\", {\n      text: \"Previously failed files\"\n    });\n    let failed_list = containerEl.createEl(\"div\");\n    this.draw_failed_files_list(failed_list);\n\n    // force refresh button\n    containerEl.createEl(\"h3\", {\n      text: \"Force Refresh\"\n    });\n    new Obsidian.Setting(containerEl).setName(\"force_refresh\").setDesc(\"WARNING: DO NOT use unless you know what you are doing! This will delete all of your current embeddings from OpenAI and trigger reprocessing of your entire vault!\").addButton((button) => button.setButtonText(\"Force Refresh\").onClick(async () => {\n      // confirm\n      if (confirm(\"Are you sure you want to Force Refresh? By clicking yes you confirm that you understand the consequences of this action.\")) {\n        // force refresh\n        await this.plugin.force_refresh_embeddings_file();\n      }\n    }));\n\n  }\n  get_self_ref_list() {\n    return \"Current: \" + SMART_TRANSLATION[this.plugin.settings.language].pronous.join(\", \");\n  }\n\n  draw_failed_files_list(failed_list) {\n    failed_list.empty();\n    if(this.plugin.settings.failed_files.length > 0) {\n      // add message that these files will be skipped until manually retried\n      failed_list.createEl(\"p\", {\n        text: \"The following files failed to process and will be skipped until manually retried.\"\n      });\n      let list = failed_list.createEl(\"ul\");\n      for (let failed_file of this.plugin.settings.failed_files) {\n        list.createEl(\"li\", {\n          text: failed_file\n        });\n      }\n      // add button to retry failed files only\n      new Obsidian.Setting(failed_list).setName(\"retry_failed_files\").setDesc(\"Retry failed files only\").addButton((button) => button.setButtonText(\"Retry failed files only\").onClick(async () => {\n        // clear failed_list element\n        failed_list.empty();\n        // set \"retrying\" text\n        failed_list.createEl(\"p\", {\n          text: \"Retrying failed files...\"\n        });\n        await this.plugin.retry_failed_files();\n        // redraw failed files list\n        this.draw_failed_files_list(failed_list);\n      }));\n    }else{\n      failed_list.createEl(\"p\", {\n        text: \"No failed files\"\n      });\n    }\n  }\n}\n\nfunction line_is_heading(line) {\n  return (line.indexOf(\"#\") === 0) && (['#', ' '].indexOf(line[1]) !== -1);\n}\n\nconst SMART_CONNECTIONS_CHAT_VIEW_TYPE = \"smart-connections-chat-view\";\n\nclass SmartConnectionsChatView extends Obsidian.ItemView {\n  constructor(leaf, plugin) {\n    super(leaf);\n    this.plugin = plugin;\n    this.active_elm = null;\n    this.active_stream = null;\n    this.brackets_ct = 0;\n    this.chat = null;\n    this.chat_box = null;\n    this.chat_container = null;\n    this.current_chat_ml = [];\n    this.files = [];\n    this.last_from = null;\n    this.message_container = null;\n    this.prevent_input = false;\n  }\n  getDisplayText() {\n    return \"Smart Connections Chat\";\n  }\n  getIcon() {\n    return \"message-square\";\n  }\n  getViewType() {\n    return SMART_CONNECTIONS_CHAT_VIEW_TYPE;\n  }\n  onOpen() {\n    this.new_chat();\n    this.plugin.get_all_folders(); // sets this.plugin.folders necessary for folder-context\n  }\n  onClose() {\n    this.chat.save_chat();\n    this.app.workspace.unregisterHoverLinkSource(SMART_CONNECTIONS_CHAT_VIEW_TYPE);\n  }\n  render_chat() {\n    this.containerEl.empty();\n    this.chat_container = this.containerEl.createDiv(\"sc-chat-container\");\n    // render plus sign for clear button\n    this.render_top_bar();\n    // render chat messages container\n    this.render_chat_box();\n    // render chat input\n    this.render_chat_input();\n    this.plugin.render_brand(this.containerEl, \"chat\");\n  }\n  // render plus sign for clear button\n  render_top_bar() {\n    // create container for clear button\n    let top_bar_container = this.chat_container.createDiv(\"sc-top-bar-container\");\n    // render the name of the chat in an input box (pop content after last hyphen in chat_id)\n    let chat_name =this.chat.name();\n    let chat_name_input = top_bar_container.createEl(\"input\", {\n      attr: {\n        type: \"text\",\n        value: chat_name\n      },\n      cls: \"sc-chat-name-input\"\n    });\n    chat_name_input.addEventListener(\"change\", this.rename_chat.bind(this));\n    \n    // create button to Smart View\n    let smart_view_btn = this.create_top_bar_button(top_bar_container, \"Smart View\", \"smart-connections\");\n    smart_view_btn.addEventListener(\"click\", this.open_smart_view.bind(this));\n    // create button to save chat\n    let save_btn = this.create_top_bar_button(top_bar_container, \"Save Chat\", \"save\");\n    save_btn.addEventListener(\"click\", this.save_chat.bind(this));\n    // create button to open chat history modal\n    let history_btn = this.create_top_bar_button(top_bar_container, \"Chat History\", \"history\");\n    history_btn.addEventListener(\"click\", this.open_chat_history.bind(this));\n    // create button to start new chat\n    const new_chat_btn = this.create_top_bar_button(top_bar_container, \"New Chat\", \"plus\");\n    new_chat_btn.addEventListener(\"click\", this.new_chat.bind(this));\n  }\n  async open_chat_history() {\n    const folder = await this.app.vault.adapter.list(\".smart-connections/chats\");\n    this.files = folder.files.map((file) => {\n      return file.replace(\".smart-connections/chats/\", \"\").replace(\".json\", \"\");\n    });\n    // open chat history modal\n    if (!this.modal)\n      this.modal = new SmartConnectionsChatHistoryModal(this.app, this);\n    this.modal.open();\n  }\n\n  create_top_bar_button(top_bar_container, title, icon=null) {\n    let btn = top_bar_container.createEl(\"button\", {\n      attr: {\n        title: title\n      }\n    });\n    if(icon){\n      Obsidian.setIcon(btn, icon);\n    }else{\n      btn.innerHTML = title;\n    }\n    return btn;\n  }\n  // render new chat\n  new_chat() {\n    this.clear_chat();\n    this.render_chat();\n    // render initial message from assistant (don't use render_message to skip adding to chat history)\n    this.new_messsage_bubble(\"assistant\");\n    this.active_elm.innerHTML = '<p>' + SMART_TRANSLATION[this.plugin.settings.language].initial_message+'</p>';\n  }\n  // open a chat from the chat history modal\n  async open_chat(chat_id) {\n    this.clear_chat();\n    await this.chat.load_chat(chat_id);\n    this.render_chat();\n    for (let i = 0; i < this.chat.chat_ml.length; i++) {\n      await this.render_message(this.chat.chat_ml[i].content, this.chat.chat_ml[i].role);\n    }\n  }\n  // clear current chat state\n  clear_chat() {\n    if (this.chat) {\n      this.chat.save_chat();\n    }\n    this.chat = new SmartConnectionsChatModel(this.plugin);\n    // if this.dotdotdot_interval is not null, clear interval\n    if (this.dotdotdot_interval) {\n      clearInterval(this.dotdotdot_interval);\n    }\n    // clear current chat ml\n    this.current_chat_ml = [];\n    // update prevent input\n    this.end_stream();\n  }\n\n  rename_chat(event) {\n    let new_chat_name = event.target.value;\n    this.chat.rename_chat(new_chat_name);\n  }\n  \n  // save current chat\n  save_chat() {\n    this.chat.save_chat();\n    new Obsidian.Notice(\"[Smart Connections] Chat saved\");\n  }\n  \n  open_smart_view() {\n    this.plugin.open_view();\n  }\n  // render chat messages container\n  render_chat_box() {\n    // create container for chat messages\n    this.chat_box = this.chat_container.createDiv(\"sc-chat-box\");\n    // create container for message\n    this.message_container = this.chat_box.createDiv(\"sc-message-container\");\n  }\n  // open file suggestion modal\n  open_file_suggestion_modal() {\n    // open file suggestion modal\n    if(!this.file_selector) this.file_selector = new SmartConnectionsFileSelectModal(this.app, this);\n    this.file_selector.open();\n  }\n  // open folder suggestion modal\n  async open_folder_suggestion_modal() {\n    // open folder suggestion modal\n    if(!this.folder_selector){\n      this.folder_selector = new SmartConnectionsFolderSelectModal(this.app, this);\n    }\n    this.folder_selector.open();\n  }\n  // insert_selection from file suggestion modal\n  insert_selection(insert_text) {\n    // get caret position\n    let caret_pos = this.textarea.selectionStart;\n    // get text before caret\n    let text_before = this.textarea.value.substring(0, caret_pos);\n    // get text after caret\n    let text_after = this.textarea.value.substring(caret_pos, this.textarea.value.length);\n    // insert text\n    this.textarea.value = text_before + insert_text + text_after;\n    // set caret position\n    this.textarea.selectionStart = caret_pos + insert_text.length;\n    this.textarea.selectionEnd = caret_pos + insert_text.length;\n    // focus on textarea\n    this.textarea.focus();\n  }\n\n  // render chat textarea and button\n  render_chat_input() {\n    // create container for chat input\n    let chat_input = this.chat_container.createDiv(\"sc-chat-form\");\n    // create textarea\n    this.textarea = chat_input.createEl(\"textarea\", {\n      cls: \"sc-chat-input\",\n      attr: {\n        placeholder: `Try \"Based on my notes\" or \"Summarize [[this note]]\" or \"Important tasks in /folder/\"`\n      }\n    });\n    // use contenteditable instead of textarea\n    // this.textarea = chat_input.createEl(\"div\", {cls: \"sc-chat-input\", attr: {contenteditable: true}});\n    // add event listener to listen for shift+enter\n    chat_input.addEventListener(\"keyup\", (e) => {\n      if([\"[\", \"/\"].indexOf(e.key) === -1) return; // skip if key is not [ or /\n      const caret_pos = this.textarea.selectionStart;\n      // if key is open square bracket\n      if (e.key === \"[\") {\n        // if previous char is [\n        if(this.textarea.value[caret_pos - 2] === \"[\"){\n          // open file suggestion modal\n          this.open_file_suggestion_modal();\n          return;\n        }\n      }else{\n        this.brackets_ct = 0;\n      }\n      // if / is pressed\n      if (e.key === \"/\") {\n        // get caret position\n        // if this is first char or previous char is space\n        if (this.textarea.value.length === 1 || this.textarea.value[caret_pos - 2] === \" \") {\n          // open folder suggestion modal\n          this.open_folder_suggestion_modal();\n          return;\n        }\n      }\n\n    });\n\n    chat_input.addEventListener(\"keydown\", (e) => {\n      if (e.key === \"Enter\" && e.shiftKey) {\n        e.preventDefault();\n        if(this.prevent_input){\n          console.log(\"wait until current response is finished\");\n          new Obsidian.Notice(\"[Smart Connections] Wait until current response is finished\");\n          return;\n        }\n        // get text from textarea\n        let user_input = this.textarea.value;\n        // clear textarea\n        this.textarea.value = \"\";\n        // initiate response from assistant\n        this.initialize_response(user_input);\n      }\n      this.textarea.style.height = 'auto';\n      this.textarea.style.height = (this.textarea.scrollHeight) + 'px';\n    });\n    // button container\n    let button_container = chat_input.createDiv(\"sc-button-container\");\n    // create hidden abort button\n    let abort_button = button_container.createEl(\"span\", { attr: {id: \"sc-abort-button\", style: \"display: none;\"} });\n    Obsidian.setIcon(abort_button, \"square\");\n    // add event listener to button\n    abort_button.addEventListener(\"click\", () => {\n      // abort current response\n      this.end_stream();\n    });\n    // create button\n    let button = button_container.createEl(\"button\", { attr: {id: \"sc-send-button\"}, cls: \"send-button\" });\n    button.innerHTML = \"Send\";\n    // add event listener to button\n    button.addEventListener(\"click\", () => {\n      if(this.prevent_input){\n        console.log(\"wait until current response is finished\");\n        new Obsidian.Notice(\"Wait until current response is finished\");\n        return;\n      }\n      // get text from textarea\n      let user_input = this.textarea.value;\n      // clear textarea\n      this.textarea.value = \"\";\n      // initiate response from assistant\n      this.initialize_response(user_input);\n    });\n  }\n  async initialize_response(user_input) {\n    this.set_streaming_ux();\n    // render message\n    await this.render_message(user_input, \"user\");\n    this.chat.new_message_in_thread({\n      role: \"user\",\n      content: user_input\n    });\n    await this.render_dotdotdot();\n\n    // if contains internal link represented by [[link]]\n    if(this.chat.contains_internal_link(user_input)) {\n      this.chat.get_response_with_note_context(user_input, this);\n      return;\n    }\n    // // for testing purposes\n    // if(this.chat.contains_folder_reference(user_input)) {\n    //   const folders = this.chat.get_folder_references(user_input);\n    //   console.log(folders);\n    //   return;\n    // }\n    // if contains self referential keywords or folder reference\n    if(this.contains_self_referential_keywords(user_input) || this.chat.contains_folder_reference(user_input)) {\n      // get hyde\n      const context = await this.get_context_hyde(user_input);\n      // get user input with added context\n      // const context_input = this.build_context_input(context);\n      // console.log(context_input);\n      const chatml = [\n        {\n          role: \"system\",\n          // content: context_input\n          content: context\n        },\n        {\n          role: \"user\",\n          content: user_input\n        }\n      ];\n      this.request_chatgpt_completion({messages: chatml, temperature: 0});\n      return;\n    }\n    // completion without any specific context\n    this.request_chatgpt_completion();\n  }\n  \n  async render_dotdotdot() {\n    if (this.dotdotdot_interval)\n      clearInterval(this.dotdotdot_interval);\n    await this.render_message(\"...\", \"assistant\");\n    // if is '...', then initiate interval to change to '.' and then to '..' and then to '...'\n    let dots = 0;\n    this.active_elm.innerHTML = '...';\n    this.dotdotdot_interval = setInterval(() => {\n      dots++;\n      if (dots > 3)\n        dots = 1;\n      this.active_elm.innerHTML = '.'.repeat(dots);\n    }, 500);\n    // wait 2 seconds for testing\n    // await new Promise(r => setTimeout(r, 2000));\n  }\n\n  set_streaming_ux() {\n    this.prevent_input = true;\n    // hide send button\n    if(document.getElementById(\"sc-send-button\"))\n      document.getElementById(\"sc-send-button\").style.display = \"none\";\n    // show abort button\n    if(document.getElementById(\"sc-abort-button\"))\n      document.getElementById(\"sc-abort-button\").style.display = \"block\";\n  }\n  unset_streaming_ux() {\n    this.prevent_input = false;\n    // show send button, remove display none\n    if(document.getElementById(\"sc-send-button\"))\n      document.getElementById(\"sc-send-button\").style.display = \"\";\n    // hide abort button\n    if(document.getElementById(\"sc-abort-button\"))\n      document.getElementById(\"sc-abort-button\").style.display = \"none\";\n  }\n\n\n  // check if includes keywords referring to one's own notes\n  contains_self_referential_keywords(user_input) {\n    const matches = user_input.match(this.plugin.self_ref_kw_regex);\n    if(matches) return true;\n    return false;\n  }\n\n  // render message\n  async render_message(message, from=\"assistant\", append_last=false) {\n    // if dotdotdot interval is set, then clear it\n    if(this.dotdotdot_interval) {\n      clearInterval(this.dotdotdot_interval);\n      this.dotdotdot_interval = null;\n      // clear last message\n      this.active_elm.innerHTML = '';\n    }\n    if(append_last) {\n      this.current_message_raw += message;\n      if(message.indexOf('\\n') === -1) {\n        this.active_elm.innerHTML += message;\n      }else{\n        this.active_elm.innerHTML = '';\n        // append to last message\n        await Obsidian.MarkdownRenderer.renderMarkdown(this.current_message_raw, this.active_elm, '?no-dataview', new Obsidian.Component());\n      }\n    }else{\n      this.current_message_raw = '';\n      if((this.chat.thread.length === 0) || (this.last_from !== from)) {\n        // create message\n        this.new_messsage_bubble(from);\n      }\n      // set message text\n      this.active_elm.innerHTML = '';\n      await Obsidian.MarkdownRenderer.renderMarkdown(message, this.active_elm, '?no-dataview', new Obsidian.Component());\n      // get links\n      this.handle_links_in_message();\n      // render button(s)\n      this.render_message_action_buttons(message);\n    }\n    // scroll to bottom\n    this.message_container.scrollTop = this.message_container.scrollHeight;\n  }\n  render_message_action_buttons(message) {\n    if (this.chat.context && this.chat.hyd) {\n      // render button to copy hyd in smart-connections code block\n      const context_view = this.active_elm.createEl(\"span\", {\n        cls: \"sc-msg-button\",\n        attr: {\n          title: \"Copy context to clipboard\" /* tooltip */\n        }\n      });\n      const this_hyd = this.chat.hyd;\n      Obsidian.setIcon(context_view, \"eye\");\n      context_view.addEventListener(\"click\", () => {\n        // copy to clipboard\n        navigator.clipboard.writeText(\"```smart-connections\\n\" + this_hyd + \"\\n```\\n\");\n        new Obsidian.Notice(\"[Smart Connections] Context code block copied to clipboard\");\n      });\n    }\n    if(this.chat.context) {\n      // render copy context button\n      const copy_prompt_button = this.active_elm.createEl(\"span\", {\n        cls: \"sc-msg-button\",\n        attr: {\n          title: \"Copy prompt to clipboard\" /* tooltip */\n        }\n      });\n      const this_context = this.chat.context.replace(/\\`\\`\\`/g, \"\\t```\").trimLeft();\n      Obsidian.setIcon(copy_prompt_button, \"files\");\n      copy_prompt_button.addEventListener(\"click\", () => {\n        // copy to clipboard\n        navigator.clipboard.writeText(\"```prompt-context\\n\" + this_context + \"\\n```\\n\");\n        new Obsidian.Notice(\"[Smart Connections] Context copied to clipboard\");\n      });\n    }\n    // render copy button\n    const copy_button = this.active_elm.createEl(\"span\", {\n      cls: \"sc-msg-button\",\n      attr: {\n        title: \"Copy message to clipboard\" /* tooltip */\n      }\n    });\n    Obsidian.setIcon(copy_button, \"copy\");\n    copy_button.addEventListener(\"click\", () => {\n      // copy message to clipboard\n      navigator.clipboard.writeText(message.trimLeft());\n      new Obsidian.Notice(\"[Smart Connections] Message copied to clipboard\");\n    });\n  }\n\n  handle_links_in_message() {\n    const links = this.active_elm.querySelectorAll(\"a\");\n    // if this active element contains a link\n    if (links.length > 0) {\n      for (let i = 0; i < links.length; i++) {\n        const link = links[i];\n        const link_text = link.getAttribute(\"data-href\");\n        // trigger hover event on link\n        link.addEventListener(\"mouseover\", (event) => {\n          this.app.workspace.trigger(\"hover-link\", {\n            event,\n            source: SMART_CONNECTIONS_CHAT_VIEW_TYPE,\n            hoverParent: link.parentElement,\n            targetEl: link,\n            // extract link text from a.data-href\n            linktext: link_text\n          });\n        });\n        // trigger open link event on link\n        link.addEventListener(\"click\", (event) => {\n          const link_tfile = this.app.metadataCache.getFirstLinkpathDest(link_text, \"/\");\n          // properly handle if the meta/ctrl key is pressed\n          const mod = Obsidian.Keymap.isModEvent(event);\n          // get most recent leaf\n          let leaf = this.app.workspace.getLeaf(mod);\n          leaf.openFile(link_tfile);\n        });\n      }\n    }\n  }\n\n  new_messsage_bubble(from) {\n    let message_el = this.message_container.createDiv(`sc-message ${from}`);\n    // create message content\n    this.active_elm = message_el.createDiv(\"sc-message-content\");\n    // set last from\n    this.last_from = from;\n  }\n\n  async request_chatgpt_completion(opts={}) {\n    const chat_ml = opts.messages || opts.chat_ml || this.chat.prepare_chat_ml();\n    console.log(\"chat_ml\", chat_ml);\n    const max_total_tokens = Math.round(get_max_chars(this.plugin.settings.smart_chat_model) / 4);\n    console.log(\"max_total_tokens\", max_total_tokens);\n    const curr_token_est = Math.round(JSON.stringify(chat_ml).length / 3);\n    console.log(\"curr_token_est\", curr_token_est);\n    let max_available_tokens = max_total_tokens - curr_token_est;\n    // if max_available_tokens is less than 0, set to 200\n    if(max_available_tokens < 0) max_available_tokens = 200;\n    else if(max_available_tokens > 4096) max_available_tokens = 4096;\n    console.log(\"max_available_tokens\", max_available_tokens);\n    opts = {\n      model: this.plugin.settings.smart_chat_model,\n      messages: chat_ml,\n      // max_tokens: 250,\n      max_tokens: max_available_tokens,\n      temperature: 0.3,\n      top_p: 1,\n      presence_penalty: 0,\n      frequency_penalty: 0,\n      stream: true,\n      stop: null,\n      n: 1,\n      // logit_bias: logit_bias,\n      ...opts\n    }\n    // console.log(opts.messages);\n    if(opts.stream) {\n      const full_str = await new Promise((resolve, reject) => {\n        try {\n          // console.log(\"stream\", opts);\n          const url = \"https://api.openai.com/v1/chat/completions\";\n          this.active_stream = new ScStreamer(url, {\n            headers: {\n              \"Content-Type\": \"application/json\",\n              Authorization: `Bearer ${this.plugin.settings.api_key}`\n            },\n            method: \"POST\",\n            payload: JSON.stringify(opts)\n          });\n          let txt = \"\";\n          this.active_stream.addEventListener(\"message\", (e) => {\n            if (e.data != \"[DONE]\") {\n              const payload = JSON.parse(e.data);\n              const text = payload.choices[0].delta.content;\n              if (!text) {\n                return;\n              }\n              txt += text;\n              this.render_message(text, \"assistant\", true);\n            } else {\n              this.end_stream();\n              resolve(txt);\n            }\n          });\n          this.active_stream.addEventListener(\"readystatechange\", (e) => {\n            if (e.readyState >= 2) {\n              console.log(\"ReadyState: \" + e.readyState);\n            }\n          });\n          this.active_stream.addEventListener(\"error\", (e) => {\n            console.error(e);\n            new Obsidian.Notice(\"Smart Connections Error Streaming Response. See console for details.\");\n            this.render_message(\"*API Error. See console logs for details.*\", \"assistant\");\n            this.end_stream();\n            reject(e);\n          });\n          this.active_stream.stream();\n        } catch (err) {\n          console.error(err);\n          new Obsidian.Notice(\"Smart Connections Error Streaming Response. See console for details.\");\n          this.end_stream();\n          reject(err);\n        }\n      });\n      // console.log(full_str);\n      await this.render_message(full_str, \"assistant\");\n      this.chat.new_message_in_thread({\n        role: \"assistant\",\n        content: full_str\n      });\n      return;\n    }else{\n      try{\n        const response = await (0, Obsidian.requestUrl)({\n          url: `https://api.openai.com/v1/chat/completions`,\n          method: \"POST\",\n          headers: {\n            Authorization: `Bearer ${this.plugin.settings.api_key}`,\n            \"Content-Type\": \"application/json\"\n          },\n          contentType: \"application/json\",\n          body: JSON.stringify(opts),\n          throw: false\n        });\n        // console.log(response);\n        return JSON.parse(response.text).choices[0].message.content;\n      }catch(err){\n        new Obsidian.Notice(`Smart Connections API Error :: ${err}`);\n      }\n    }\n  }\n\n  end_stream() {\n    if(this.active_stream){\n      this.active_stream.close();\n      this.active_stream = null;\n    }\n    this.unset_streaming_ux();\n    if(this.dotdotdot_interval){\n      clearInterval(this.dotdotdot_interval);\n      this.dotdotdot_interval = null;\n      // remove parent of active_elm\n      this.active_elm.parentElement.remove();\n      this.active_elm = null;\n    }\n  }\n\n  async get_context_hyde(user_input) {\n    this.chat.reset_context();\n    // count current chat ml messages to determine 'question' or 'chat log' wording\n    const hyd_input = `Anticipate what the user is seeking. Respond in the form of a hypothetical note written by the user. The note may contain statements as paragraphs, lists, or checklists in markdown format with no headings. Please respond with one hypothetical note and abstain from any other commentary. Use the format: PARENT FOLDER NAME > CHILD FOLDER NAME > FILE NAME > HEADING 1 > HEADING 2 > HEADING 3: HYPOTHETICAL NOTE CONTENTS.`;\n    // complete\n    const chatml = [\n      {\n        role: \"system\",\n        content: hyd_input \n      },\n      {\n        role: \"user\",\n        content: user_input\n      }\n    ];\n    const hyd = await this.request_chatgpt_completion({\n      messages: chatml,\n      stream: false,\n      temperature: 0,\n      max_tokens: 137,\n    });\n    this.chat.hyd = hyd;\n    // console.log(hyd);\n    let filter = {};\n    // if contains folder reference represented by /folder/\n    if(this.chat.contains_folder_reference(user_input)) {\n      // get folder references\n      const folder_refs = this.chat.get_folder_references(user_input);\n      // console.log(folder_refs);\n      // if folder references are valid (string or array of strings)\n      if(folder_refs){\n        filter = {\n          path_begins_with: folder_refs\n        };\n      }\n    }\n    // search for nearest based on hyd\n    let nearest = await this.plugin.api.search(hyd, filter);\n    console.log(\"nearest\", nearest.length);\n    nearest = this.get_nearest_until_next_dev_exceeds_std_dev(nearest);\n    console.log(\"nearest after std dev slice\", nearest.length);\n    nearest = this.sort_by_len_adjusted_similarity(nearest);\n    \n    return await this.get_context_for_prompt(nearest);\n  }\n  \n  \n  sort_by_len_adjusted_similarity(nearest) {\n    // re-sort by quotient of similarity divided by len DESC\n    nearest = nearest.sort((a, b) => {\n      const a_score = a.similarity / a.len;\n      const b_score = b.similarity / b.len;\n      // if a is greater than b, return -1\n      if (a_score > b_score)\n        return -1;\n      // if a is less than b, return 1\n      if (a_score < b_score)\n        return 1;\n      // if a is equal to b, return 0\n      return 0;\n    });\n    return nearest;\n  }\n\n  get_nearest_until_next_dev_exceeds_std_dev(nearest) {\n    // get std dev of similarity\n    const sim = nearest.map((n) => n.similarity);\n    const mean = sim.reduce((a, b) => a + b) / sim.length;\n    let std_dev = Math.sqrt(sim.map((x) => Math.pow(x - mean, 2)).reduce((a, b) => a + b) / sim.length);\n    // slice where next item deviation is greater than std_dev\n    let slice_i = 0;\n    while (slice_i < nearest.length) {\n      const next = nearest[slice_i + 1];\n      if (next) {\n        const next_dev = Math.abs(next.similarity - nearest[slice_i].similarity);\n        if (next_dev > std_dev) {\n          if(slice_i < 3) std_dev = std_dev * 1.5;\n          else break;\n        }\n      }\n      slice_i++;\n    }\n    // select top results\n    nearest = nearest.slice(0, slice_i+1);\n    return nearest;\n  }\n  // this.test_get_nearest_until_next_dev_exceeds_std_dev();\n  // // test get_nearest_until_next_dev_exceeds_std_dev\n  // test_get_nearest_until_next_dev_exceeds_std_dev() {\n  //   const nearest = [{similarity: 0.99}, {similarity: 0.98}, {similarity: 0.97}, {similarity: 0.96}, {similarity: 0.95}, {similarity: 0.94}, {similarity: 0.93}, {similarity: 0.92}, {similarity: 0.91}, {similarity: 0.9}, {similarity: 0.79}, {similarity: 0.78}, {similarity: 0.77}, {similarity: 0.76}, {similarity: 0.75}, {similarity: 0.74}, {similarity: 0.73}, {similarity: 0.72}];\n  //   const result = this.get_nearest_until_next_dev_exceeds_std_dev(nearest);\n  //   if(result.length !== 10){\n  //     console.error(\"get_nearest_until_next_dev_exceeds_std_dev failed\", result);\n  //   }\n  // }\n\n  async get_context_for_prompt(nearest) {\n    let context = [];\n    const MAX_SOURCES = (this.plugin.settings.smart_chat_model === 'gpt-4-1106-preview') ? 42 : 20;\n    const MAX_CHARS = get_max_chars(this.plugin.settings.smart_chat_model) / 2;\n    let char_accum = 0;\n    for (let i = 0; i < nearest.length; i++) {\n      if (context.length >= MAX_SOURCES)\n        break;\n      if (char_accum >= MAX_CHARS)\n        break;\n      if (typeof nearest[i].link !== 'string')\n        continue;\n      // generate breadcrumbs\n      const breadcrumbs = nearest[i].link.replace(/#/g, \" > \").replace(\".md\", \"\").replace(/\\//g, \" > \");\n      let new_context = `${breadcrumbs}:\\n`;\n      // get max available chars to add to context\n      const max_available_chars = MAX_CHARS - char_accum - new_context.length;\n      if (nearest[i].link.indexOf(\"#\") !== -1) { // is block\n        new_context += await this.plugin.block_retriever(nearest[i].link, { max_chars: max_available_chars });\n      } else { // is file\n        new_context += await this.plugin.file_retriever(nearest[i].link, { max_chars: max_available_chars });\n      }\n      // add to char_accum\n      char_accum += new_context.length;\n      // add to context\n      context.push({\n        link: nearest[i].link,\n        text: new_context\n      });\n    }\n    // context sources\n    console.log(\"context sources: \" + context.length);\n    // char_accum divided by 4 and rounded to nearest integer for estimated tokens\n    console.log(\"total context tokens: ~\" + Math.round(char_accum / 3.5));\n    // build context input\n    this.chat.context = `Anticipate the type of answer desired by the user. Imagine the following ${context.length} notes were written by the user and contain all the necessary information to answer the user's question. Begin responses with \"${SMART_TRANSLATION[this.plugin.settings.language].prompt}...\"`;\n    for(let i = 0; i < context.length; i++) {\n      this.chat.context += `\\n---BEGIN #${i+1}---\\n${context[i].text}\\n---END #${i+1}---`;\n    }\n    return this.chat.context;\n  }\n\n\n}\n\nfunction get_max_chars(model=\"gpt-3.5-turbo\") {\n  const MAX_CHAR_MAP = {\n    \"gpt-3.5-turbo-16k\": 48000,\n    \"gpt-4\": 24000,\n    \"gpt-3.5-turbo\": 12000,\n    \"gpt-4-1106-preview\": 200000,\n  };\n  return MAX_CHAR_MAP[model];\n}\n/**\n * SmartConnectionsChatModel\n * ---\n * - 'thread' format: Array[Array[Object{role, content, hyde}]]\n *  - [Turn[variation{}], Turn[variation{}, variation{}], ...]\n * - Saves in 'thread' format to JSON file in .smart-connections folder using chat_id as filename\n * - Loads chat in 'thread' format Array[Array[Object{role, content, hyde}]] from JSON file in .smart-connections folder\n * - prepares chat_ml returns in 'ChatML' format \n *  - strips all but role and content properties from Object in ChatML format\n * - ChatML Array[Object{role, content}]\n *  - [Current_Variation_For_Turn_1{}, Current_Variation_For_Turn_2{}, ...]\n */\nclass SmartConnectionsChatModel {\n  constructor(plugin) {\n    this.app = plugin.app;\n    this.plugin = plugin;\n    this.chat_id = null;\n    this.chat_ml = [];\n    this.context = null;\n    this.hyd = null;\n    this.thread = [];\n  }\n  async save_chat() {\n    // return if thread is empty\n    if (this.thread.length === 0) return;\n    // save chat to file in .smart-connections folder\n    // create .smart-connections/chats/ folder if it doesn't exist\n    if (!(await this.app.vault.adapter.exists(\".smart-connections/chats\"))) {\n      await this.app.vault.adapter.mkdir(\".smart-connections/chats\");\n    }\n    // if chat_id is not set, set it to UNTITLED-${unix timestamp}\n    if (!this.chat_id) {\n      this.chat_id = this.name() + \"\u2014\" + this.get_file_date_string();\n    }\n    // validate chat_id is set to valid filename characters (letters, numbers, underscores, dashes, em dash, and spaces)\n    if (!this.chat_id.match(/^[a-zA-Z0-9_\u2014\\- ]+$/)) {\n      console.log(\"Invalid chat_id: \" + this.chat_id);\n      new Obsidian.Notice(\"[Smart Connections] Failed to save chat. Invalid chat_id: '\" + this.chat_id + \"'\");\n    }\n    // filename is chat_id\n    const chat_file = this.chat_id + \".json\";\n    this.app.vault.adapter.write(\n      \".smart-connections/chats/\" + chat_file,\n      JSON.stringify(this.thread, null, 2)\n    );\n  }\n  async load_chat(chat_id) {\n    this.chat_id = chat_id;\n    // load chat from file in .smart-connections folder\n    // filename is chat_id\n    const chat_file = this.chat_id + \".json\";\n    // read file\n    let chat_json = await this.app.vault.adapter.read(\n      \".smart-connections/chats/\" + chat_file\n    );\n    // parse json\n    this.thread = JSON.parse(chat_json);\n    // load chat_ml\n    this.chat_ml = this.prepare_chat_ml();\n    // render messages in chat view\n    // for each turn in chat_ml\n    // console.log(this.thread);\n    // console.log(this.chat_ml);\n  }\n  // prepare chat_ml from chat\n  // gets the last message of each turn unless turn_variation_offsets=[[turn_index,variation_index]] is specified in offset\n  prepare_chat_ml(turn_variation_offsets=[]) {\n    // if no turn_variation_offsets, get the last message of each turn\n    if(turn_variation_offsets.length === 0){\n      this.chat_ml = this.thread.map(turn => {\n        return turn[turn.length - 1];\n      });\n    }else{\n      // create an array from turn_variation_offsets that indexes variation_index at turn_index\n      // ex. [[3,5]] => [undefined, undefined, undefined, 5]\n      let turn_variation_index = [];\n      for(let i = 0; i < turn_variation_offsets.length; i++){\n        turn_variation_index[turn_variation_offsets[i][0]] = turn_variation_offsets[i][1];\n      }\n      // loop through chat\n      this.chat_ml = this.thread.map((turn, turn_index) => {\n        // if there is an index for this turn, return the variation at that index\n        if(turn_variation_index[turn_index] !== undefined){\n          return turn[turn_variation_index[turn_index]];\n        }\n        // otherwise return the last message of the turn\n        return turn[turn.length - 1];\n      });\n    }\n    // strip all but role and content properties from each message\n    this.chat_ml = this.chat_ml.map(message => {\n      return {\n        role: message.role,\n        content: message.content\n      };\n    });\n    return this.chat_ml;\n  }\n  last() {\n    // get last message from chat\n    return this.thread[this.thread.length - 1][this.thread[this.thread.length - 1].length - 1];\n  }\n  last_from() {\n    return this.last().role;\n  }\n  // returns user_input or completion\n  last_message() {\n    return this.last().content;\n  }\n  // message={}\n  // add new message to thread\n  new_message_in_thread(message, turn=-1) {\n    // if turn is -1, add to new turn\n    if(this.context){\n      message.context = this.context;\n      this.context = null;\n    }\n    if(this.hyd){\n      message.hyd = this.hyd;\n      this.hyd = null;\n    }\n    if (turn === -1) {\n      this.thread.push([message]);\n    }else{\n      // otherwise add to specified turn\n      this.thread[turn].push(message);\n    }\n  }\n  reset_context(){\n    this.context = null;\n    this.hyd = null;\n  }\n  async rename_chat(new_name){\n    // check if current chat_id file exists\n    if (this.chat_id && await this.app.vault.adapter.exists(\".smart-connections/chats/\" + this.chat_id + \".json\")) {\n      new_name = this.chat_id.replace(this.name(), new_name);\n      // rename file if it exists\n      await this.app.vault.adapter.rename(\n        \".smart-connections/chats/\" + this.chat_id + \".json\",\n        \".smart-connections/chats/\" + new_name + \".json\"\n      );\n      // set chat_id to new_name\n      this.chat_id = new_name;\n    }else{\n      this.chat_id = new_name + \"\u2014\" + this.get_file_date_string();\n      // save chat\n      await this.save_chat();\n    }\n\n  }\n\n  name() {\n    if(this.chat_id){\n      // remove date after last em dash\n      return this.chat_id.replace(/\u2014[^\u2014]*$/,\"\");\n    }\n    return \"UNTITLED\";\n  }\n\n  get_file_date_string() {\n    return new Date().toISOString().replace(/(T|:|\\..*)/g, \" \").trim();\n  }\n  // get response from with note context\n  async get_response_with_note_context(user_input, chat_view) {\n    let system_input = \"Imagine the following notes were written by the user and contain the necessary information to synthesize a useful answer the user's query:\\n\";\n    // extract internal links\n    const notes = this.extract_internal_links(user_input);\n    // get content of internal links as context\n    let max_chars = get_max_chars(this.plugin.settings.smart_chat_model);\n    for(let i = 0; i < notes.length; i++){\n      // max chars for this note is max_chars divided by number of notes left\n      const this_max_chars = (notes.length - i > 1) ? Math.floor(max_chars / (notes.length - i)) : max_chars;\n      // console.log(\"file context max chars: \" + this_max_chars);\n      const note_content = await this.get_note_contents(notes[i], {char_limit: this_max_chars});\n      system_input += `---BEGIN NOTE: [[${notes[i].basename}]]---\\n`\n      system_input += note_content;\n      system_input += `---END NOTE---\\n`\n      max_chars -= note_content.length;\n      if(max_chars <= 0) break;\n    }\n    this.context = system_input;\n    const chatml = [\n      {\n        role: \"system\",\n        content: system_input\n      },\n      {\n        role: \"user\",\n        content: user_input\n      }\n    ];\n    chat_view.request_chatgpt_completion({messages: chatml, temperature: 0});\n  }\n  // check if contains internal link\n  contains_internal_link(user_input) {\n    if(user_input.indexOf(\"[[\") === -1) return false;\n    if(user_input.indexOf(\"]]\") === -1) return false;\n    return true;\n  }\n  // check if contains folder reference (ex. /folder/, or /folder/subfolder/)\n  contains_folder_reference(user_input) {\n    if(user_input.indexOf(\"/\") === -1) return false;\n    if(user_input.indexOf(\"/\") === user_input.lastIndexOf(\"/\")) return false;\n    return true;\n  }\n  // get folder references from user input\n  get_folder_references(user_input) {\n    // use this.folders to extract folder references by longest first (ex. /folder/subfolder/ before /folder/) to avoid matching /folder/subfolder/ as /folder/\n    const folders = this.plugin.folders.slice(); // copy folders array\n    const matches = folders.sort((a, b) => b.length - a.length).map(folder => {\n      // check if folder is in user_input\n      if(user_input.indexOf(folder) !== -1){\n        // remove folder from user_input to prevent matching /folder/subfolder/ as /folder/\n        user_input = user_input.replace(folder, \"\");\n        return folder;\n      }\n      return false;\n    }).filter(folder => folder);\n    console.log(matches);\n    // return array of matches\n    if(matches) return matches;\n    return false;\n  }\n\n\n  // extract internal links\n  extract_internal_links(user_input) {\n    const matches = user_input.match(/\\[\\[(.*?)\\]\\]/g);\n    console.log(matches);\n    // return array of TFile objects\n    if(matches) return matches.map(match => {\n      return this.app.metadataCache.getFirstLinkpathDest(match.replace(\"[[\", \"\").replace(\"]]\", \"\"), \"/\");\n    });\n    return [];\n  }\n  // get context from internal links\n  async get_note_contents(note, opts={}) {\n    opts = {\n      char_limit: 10000,\n      ...opts\n    }\n    // return if note is not a file\n    if(!(note instanceof Obsidian.TFile)) return \"\";\n    // get file content\n    let file_content = await this.app.vault.cachedRead(note);\n    // check if contains dataview code block\n    if(file_content.indexOf(\"```dataview\") > -1){\n      // if contains dataview code block get all dataview code blocks\n      file_content = await this.render_dataview_queries(file_content, note.path, opts);\n    }\n    file_content = file_content.substring(0, opts.char_limit);\n    // console.log(file_content.length);\n    return file_content;\n  }\n\n\n  async render_dataview_queries(file_content, note_path, opts={}) {\n    opts = {\n      char_limit: null,\n      ...opts\n    };\n    // use window to get dataview api\n    const dataview_api = window[\"DataviewAPI\"];\n    // skip if dataview api not found\n    if(!dataview_api) return file_content;\n    const dataview_code_blocks = file_content.match(/```dataview(.*?)```/gs);\n    // for each dataview code block\n    for (let i = 0; i < dataview_code_blocks.length; i++) {\n      // if opts char_limit is less than indexOf dataview code block, break\n      if(opts.char_limit && opts.char_limit < file_content.indexOf(dataview_code_blocks[i])) break;\n      // get dataview code block\n      const dataview_code_block = dataview_code_blocks[i];\n      // get content of dataview code block\n      const dataview_code_block_content = dataview_code_block.replace(\"```dataview\", \"\").replace(\"```\", \"\");\n      // get dataview query result\n      const dataview_query_result = await dataview_api.queryMarkdown(dataview_code_block_content, note_path, null);\n      // if query result is successful, replace dataview code block with query result\n      if (dataview_query_result.successful) {\n        file_content = file_content.replace(dataview_code_block, dataview_query_result.value);\n      }\n    }\n    return file_content;\n  }\n}\n\nclass SmartConnectionsChatHistoryModal extends Obsidian.FuzzySuggestModal {\n  constructor(app, view, files) {\n    super(app);\n    this.app = app;\n    this.view = view;\n    this.setPlaceholder(\"Type the name of a chat session...\");\n  }\n  getItems() {\n    if (!this.view.files) {\n      return [];\n    }\n    return this.view.files;\n  }\n  getItemText(item) {\n    // if not UNTITLED, remove date after last em dash\n    if(item.indexOf(\"UNTITLED\") === -1){\n      item.replace(/\u2014[^\u2014]*$/,\"\");\n    }\n    return item;\n  }\n  onChooseItem(session) {\n    this.view.open_chat(session);\n  }\n}\n\n// File Select Fuzzy Suggest Modal\nclass SmartConnectionsFileSelectModal extends Obsidian.FuzzySuggestModal {\n  constructor(app, view) {\n    super(app);\n    this.app = app;\n    this.view = view;\n    this.setPlaceholder(\"Type the name of a file...\");\n  }\n  getItems() {\n    // get all markdown files\n    return this.app.vault.getMarkdownFiles().sort((a, b) => a.basename.localeCompare(b.basename));\n  }\n  getItemText(item) {\n    return item.basename;\n  }\n  onChooseItem(file) {\n    this.view.insert_selection(file.basename + \"]] \");\n  }\n}\n// Folder Select Fuzzy Suggest Modal\nclass SmartConnectionsFolderSelectModal extends Obsidian.FuzzySuggestModal {\n  constructor(app, view) {\n    super(app);\n    this.app = app;\n    this.view = view;\n    this.setPlaceholder(\"Type the name of a folder...\");\n  }\n  getItems() {\n    return this.view.plugin.folders;\n  }\n  getItemText(item) {\n    return item;\n  }\n  onChooseItem(folder) {\n    this.view.insert_selection(folder + \"/ \");\n  }\n}\n\n\n// Handle API response streaming\nclass ScStreamer {\n  // constructor\n  constructor(url, options) {\n    // set default options\n    options = options || {};\n    this.url = url;\n    this.method = options.method || 'GET';\n    this.headers = options.headers || {};\n    this.payload = options.payload || null;\n    this.withCredentials = options.withCredentials || false;\n    this.listeners = {};\n    this.readyState = this.CONNECTING;\n    this.progress = 0;\n    this.chunk = '';\n    this.xhr = null;\n    this.FIELD_SEPARATOR = ':';\n    this.INITIALIZING = -1;\n    this.CONNECTING = 0;\n    this.OPEN = 1;\n    this.CLOSED = 2;\n  }\n  // addEventListener\n  addEventListener(type, listener) {\n    // check if the type is in the listeners\n    if (!this.listeners[type]) {\n      this.listeners[type] = [];\n    }\n    // check if the listener is already in the listeners\n    if(this.listeners[type].indexOf(listener) === -1) {\n      this.listeners[type].push(listener);\n    }\n  }\n  // removeEventListener\n  removeEventListener(type, listener) {\n    // check if listener type is undefined\n    if (!this.listeners[type]) {\n      return;\n    }\n    let filtered = [];\n    // loop through the listeners\n    for (let i = 0; i < this.listeners[type].length; i++) {\n      // check if the listener is the same\n      if (this.listeners[type][i] !== listener) {\n        filtered.push(this.listeners[type][i]);\n      }\n    }\n    // check if the listeners are empty\n    if (this.listeners[type].length === 0) {\n      delete this.listeners[type];\n    } else {\n      this.listeners[type] = filtered;\n    }\n  }\n  // dispatchEvent\n  dispatchEvent(event) {\n    // if no event return true\n    if (!event) {\n      return true;\n    }\n    // set event source to this\n    event.source = this;\n    // set onHandler to on + event type\n    let onHandler = 'on' + event.type;\n    // check if the onHandler has own property named same as onHandler\n    if (this.hasOwnProperty(onHandler)) {\n      // call the onHandler\n      this[onHandler].call(this, event);\n      // check if the event is default prevented\n      if (event.defaultPrevented) {\n        return false;\n      }\n    }\n    // check if the event type is in the listeners\n    if (this.listeners[event.type]) {\n      return this.listeners[event.type].every(function(callback) {\n        callback(event);\n        return !event.defaultPrevented;\n      });\n    }\n    return true;\n  }\n  // _setReadyState\n  _setReadyState(state) {\n    // set event type to readyStateChange\n    let event = new CustomEvent('readyStateChange');\n    // set event readyState to state\n    event.readyState = state;\n    // set readyState to state\n    this.readyState = state;\n    // dispatch event\n    this.dispatchEvent(event);\n  }\n  // _onStreamFailure\n  _onStreamFailure(e) {\n    // set event type to error\n    let event = new CustomEvent('error');\n    // set event data to e\n    event.data = e.currentTarget.response;\n    // dispatch event\n    this.dispatchEvent(event);\n    this.close();\n  }\n  // _onStreamAbort\n  _onStreamAbort(e) {\n    // set to abort\n    let event = new CustomEvent('abort');\n    // close\n    this.close();\n  }\n  // _onStreamProgress\n  _onStreamProgress(e) {\n    // if not xhr return\n    if (!this.xhr) {\n      return;\n    }\n    // if xhr status is not 200 return\n    if (this.xhr.status !== 200) {\n      // onStreamFailure\n      this._onStreamFailure(e);\n      return;\n    }\n    // if ready state is CONNECTING\n    if (this.readyState === this.CONNECTING) {\n      // dispatch event\n      this.dispatchEvent(new CustomEvent('open'));\n      // set ready state to OPEN\n      this._setReadyState(this.OPEN);\n    }\n    // parse the received data.\n    let data = this.xhr.responseText.substring(this.progress);\n    // update progress\n    this.progress += data.length;\n    // split the data by new line and parse each line\n    data.split(/(\\r\\n|\\r|\\n){2}/g).forEach(function(part){\n      if(part.trim().length === 0) {\n        this.dispatchEvent(this._parseEventChunk(this.chunk.trim()));\n        this.chunk = '';\n      } else {\n        this.chunk += part;\n      }\n    }.bind(this));\n  }\n  // _onStreamLoaded\n  _onStreamLoaded(e) {\n    this._onStreamProgress(e);\n    // parse the last chunk\n    this.dispatchEvent(this._parseEventChunk(this.chunk));\n    this.chunk = '';\n  }\n  // _parseEventChunk\n  _parseEventChunk(chunk) {\n    // if no chunk or chunk is empty return\n    if (!chunk || chunk.length === 0) {\n      return null;\n    }\n    // init e\n    let e = {id: null, retry: null, data: '', event: 'message'};\n    // split the chunk by new line\n    chunk.split(/(\\r\\n|\\r|\\n)/).forEach(function(line) {\n      line = line.trimRight();\n      let index = line.indexOf(this.FIELD_SEPARATOR);\n      if(index <= 0) {\n        return;\n      }\n      // field\n      let field = line.substring(0, index);\n      if(!(field in e)) {\n        return;\n      }\n      // value\n      let value = line.substring(index + 1).trimLeft();\n      if(field === 'data') {\n        e[field] += value;\n      } else {\n        e[field] = value;\n      }\n    }.bind(this));\n    // return event\n    let event = new CustomEvent(e.event);\n    event.data = e.data;\n    event.id = e.id;\n    return event;\n  }\n  // _checkStreamClosed\n  _checkStreamClosed() {\n    if(!this.xhr) {\n      return;\n    }\n    if(this.xhr.readyState === XMLHttpRequest.DONE) {\n      this._setReadyState(this.CLOSED);\n    }\n  }\n  // stream\n  stream() {\n    // set ready state to connecting\n    this._setReadyState(this.CONNECTING);\n    // set xhr to new XMLHttpRequest\n    this.xhr = new XMLHttpRequest();\n    // set xhr progress to _onStreamProgress\n    this.xhr.addEventListener('progress', this._onStreamProgress.bind(this));\n    // set xhr load to _onStreamLoaded\n    this.xhr.addEventListener('load', this._onStreamLoaded.bind(this));\n    // set xhr ready state change to _checkStreamClosed\n    this.xhr.addEventListener('readystatechange', this._checkStreamClosed.bind(this));\n    // set xhr error to _onStreamFailure\n    this.xhr.addEventListener('error', this._onStreamFailure.bind(this));\n    // set xhr abort to _onStreamAbort\n    this.xhr.addEventListener('abort', this._onStreamAbort.bind(this));\n    // open xhr\n    this.xhr.open(this.method, this.url);\n    // headers to xhr\n    for (let header in this.headers) {\n      this.xhr.setRequestHeader(header, this.headers[header]);\n    }\n    // credentials to xhr\n    this.xhr.withCredentials = this.withCredentials;\n    // send xhr\n    this.xhr.send(this.payload);\n  }\n  // close\n  close() {\n    if(this.readyState === this.CLOSED) {\n      return;\n    }\n    this.xhr.abort();\n    this.xhr = null;\n    this._setReadyState(this.CLOSED);\n  }\n}\n\nmodule.exports = SmartConnectionsPlugin;"],
  "mappings": ";;;;;;AAAA;AAAA,4BAAAA,UAAAC,SAAA;AAAA,QAAMC,WAAN,MAAc;AAAA,MACZ,YAAY,QAAQ;AAElB,aAAK,SAAS;AAAA,UACZ,WAAW;AAAA,UACX,aAAa;AAAA,UACb,gBAAgB;AAAA,UAChB,eAAe;AAAA,UACf,cAAc;AAAA,UACd,gBAAgB;AAAA,UAChB,cAAc;AAAA,UACd,eAAe;AAAA,UACf,GAAG;AAAA,QACL;AACA,aAAK,YAAY,KAAK,OAAO;AAC7B,aAAK,cAAc,OAAO;AAC1B,aAAK,YAAY,KAAK,cAAc,MAAM,KAAK;AAE/C,aAAK,aAAa;AAAA,MACpB;AAAA,MACA,MAAM,YAAY,MAAM;AACtB,YAAI,KAAK,OAAO,gBAAgB;AAC9B,iBAAO,MAAM,KAAK,OAAO,eAAe,IAAI;AAAA,QAC9C,OAAO;AAEL,gBAAM,IAAI,MAAM,wBAAwB;AAAA,QAC1C;AAAA,MACF;AAAA,MACA,MAAM,MAAM,MAAM;AAChB,YAAI,KAAK,OAAO,eAAe;AAC7B,iBAAO,MAAM,KAAK,OAAO,cAAc,IAAI;AAAA,QAC7C,OAAO;AAEL,gBAAM,IAAI,MAAM,uBAAuB;AAAA,QACzC;AAAA,MACF;AAAA,MACA,MAAM,UAAU,MAAM;AACpB,YAAI,KAAK,OAAO,cAAc;AAC5B,iBAAO,MAAM,KAAK,OAAO,aAAa,IAAI;AAAA,QAC5C,OAAO;AAEL,gBAAM,IAAI,MAAM,sBAAsB;AAAA,QACxC;AAAA,MACF;AAAA,MACA,MAAM,OAAO,UAAU,UAAU;AAC/B,YAAI,KAAK,OAAO,gBAAgB;AAC9B,iBAAO,MAAM,KAAK,OAAO,eAAe,UAAU,QAAQ;AAAA,QAC5D,OAAO;AAEL,gBAAM,IAAI,MAAM,wBAAwB;AAAA,QAC1C;AAAA,MACF;AAAA,MACA,MAAM,KAAK,MAAM;AACf,YAAI,KAAK,OAAO,cAAc;AAC5B,iBAAO,MAAM,KAAK,OAAO,aAAa,IAAI;AAAA,QAC5C,OAAO;AAEL,gBAAM,IAAI,MAAM,sBAAsB;AAAA,QACxC;AAAA,MACF;AAAA,MACA,MAAM,WAAW,MAAM,MAAM;AAC3B,YAAI,KAAK,OAAO,eAAe;AAC7B,iBAAO,MAAM,KAAK,OAAO,cAAc,MAAM,IAAI;AAAA,QACnD,OAAO;AAEL,gBAAM,IAAI,MAAM,uBAAuB;AAAA,QACzC;AAAA,MACF;AAAA,MACA,MAAM,KAAK,UAAU,GAAG;AACtB,YAAI;AACF,gBAAM,kBAAkB,MAAM,KAAK,UAAU,KAAK,SAAS;AAE3D,eAAK,aAAa,KAAK,MAAM,eAAe;AAC5C,kBAAQ,IAAI,6BAA2B,KAAK,SAAS;AACrD,iBAAO;AAAA,QACT,SAAS,OAAP;AAEA,cAAI,UAAU,GAAG;AACf,oBAAQ,IAAI,iBAAiB;AAE7B,kBAAM,IAAI,QAAQ,OAAK,WAAW,GAAG,MAAQ,MAAO,OAAQ,CAAC;AAC7D,mBAAO,MAAM,KAAK,KAAK,UAAU,CAAC;AAAA,UASpC;AACA,kBAAQ,IAAI,oEAAoE;AAChF,iBAAO;AAAA,QACT;AAAA,MACF;AAAA,MAEA,MAAM,uBAAuB;AAE3B,YAAI,CAAE,MAAM,KAAK,YAAY,KAAK,WAAW,GAAI;AAE/C,gBAAM,KAAK,MAAM,KAAK,WAAW;AACjC,kBAAQ,IAAI,qBAAmB,KAAK,WAAW;AAAA,QACjD,OAAO;AACL,kBAAQ,IAAI,4BAA0B,KAAK,WAAW;AAAA,QACxD;AAEA,YAAI,CAAE,MAAM,KAAK,YAAY,KAAK,SAAS,GAAI;AAE7C,gBAAM,KAAK,WAAW,KAAK,WAAW,IAAI;AAC1C,kBAAQ,IAAI,8BAA4B,KAAK,SAAS;AAAA,QACxD,OAAO;AACL,kBAAQ,IAAI,qCAAmC,KAAK,SAAS;AAAA,QAC/D;AAAA,MACF;AAAA,MAEA,MAAM,OAAO;AACX,cAAM,aAAa,KAAK,UAAU,KAAK,UAAU;AAEjD,cAAM,yBAAyB,MAAM,KAAK,YAAY,KAAK,SAAS;AAEpE,YAAI,wBAAwB;AAE1B,gBAAM,gBAAgB,WAAW;AAEjC,gBAAM,qBAAqB,MAAM,KAAK,KAAK,KAAK,SAAS,EAAE,KAAK,CAAC,SAAS,KAAK,IAAI;AAInF,cAAI,gBAAiB,qBAAqB,KAAM;AAE9C,kBAAM,KAAK,WAAW,KAAK,WAAW,UAAU;AAChD,oBAAQ,IAAI,2BAA2B,gBAAgB,QAAQ;AAAA,UACjE,OAAO;AAGL,kBAAM,kBAAkB;AAAA,cACtB;AAAA,cACA;AAAA,cACA,oBAAoB,gBAAgB;AAAA,cACpC,yBAAyB,qBAAqB;AAAA,cAC9C;AAAA,YACF;AACA,oBAAQ,IAAI,gBAAgB,KAAK,GAAG,CAAC;AAErC,kBAAM,KAAK,WAAW,KAAK,cAAY,4BAA4B,UAAU;AAC7E,kBAAM,IAAI,MAAM,oJAAoJ;AAAA,UACtK;AAAA,QACF,OAAO;AACL,gBAAM,KAAK,qBAAqB;AAChC,iBAAO,MAAM,KAAK,KAAK;AAAA,QACzB;AACA,eAAO;AAAA,MACT;AAAA,MACA,QAAQ,SAAS,SAAS;AACxB,YAAI,aAAa;AACjB,YAAI,QAAQ;AACZ,YAAI,QAAQ;AACZ,iBAAS,IAAI,GAAG,IAAI,QAAQ,QAAQ,KAAK;AACvC,wBAAc,QAAQ,CAAC,IAAI,QAAQ,CAAC;AACpC,mBAAS,QAAQ,CAAC,IAAI,QAAQ,CAAC;AAC/B,mBAAS,QAAQ,CAAC,IAAI,QAAQ,CAAC;AAAA,QACjC;AACA,YAAI,UAAU,KAAK,UAAU,GAAG;AAC9B,iBAAO;AAAA,QACT,OAAO;AACL,iBAAO,cAAc,KAAK,KAAK,KAAK,IAAI,KAAK,KAAK,KAAK;AAAA,QACzD;AAAA,MACF;AAAA,MACA,QAAQ,QAAQ,SAAS,CAAC,GAAG;AAC3B,iBAAS;AAAA,UACP,eAAe;AAAA,UACf,GAAG;AAAA,QACL;AACA,YAAI,UAAU,CAAC;AACf,cAAM,YAAY,OAAO,KAAK,KAAK,UAAU;AAE7C,iBAAS,IAAI,GAAG,IAAI,UAAU,QAAQ,KAAK;AAEzC,cAAI,OAAO,eAAe;AACxB,kBAAM,YAAY,KAAK,WAAW,UAAU,CAAC,CAAC,EAAE,KAAK;AACrD,gBAAI,UAAU,QAAQ,GAAG,IAAI;AAAI;AAAA,UAGnC;AACA,cAAI,OAAO,UAAU;AACnB,gBAAI,OAAO,aAAa,UAAU,CAAC;AAAG;AACtC,gBAAI,OAAO,aAAa,KAAK,WAAW,UAAU,CAAC,CAAC,EAAE,KAAK;AAAQ;AAAA,UACrE;AAEA,cAAI,OAAO,kBAAkB;AAE3B,gBAAI,OAAO,OAAO,qBAAqB,YAAY,CAAC,KAAK,WAAW,UAAU,CAAC,CAAC,EAAE,KAAK,KAAK,WAAW,OAAO,gBAAgB;AAAG;AAEjI,gBAAI,MAAM,QAAQ,OAAO,gBAAgB,KAAK,CAAC,OAAO,iBAAiB,KAAK,CAAC,SAAS,KAAK,WAAW,UAAU,CAAC,CAAC,EAAE,KAAK,KAAK,WAAW,IAAI,CAAC;AAAG;AAAA,UACnJ;AAEA,kBAAQ,KAAK;AAAA,YACX,MAAM,KAAK,WAAW,UAAU,CAAC,CAAC,EAAE,KAAK;AAAA,YACzC,YAAY,KAAK,QAAQ,QAAQ,KAAK,WAAW,UAAU,CAAC,CAAC,EAAE,GAAG;AAAA,YAClE,MAAM,KAAK,WAAW,UAAU,CAAC,CAAC,EAAE,KAAK;AAAA,UAC3C,CAAC;AAAA,QACH;AAEA,gBAAQ,KAAK,SAAU,GAAG,GAAG;AAC3B,iBAAO,EAAE,aAAa,EAAE;AAAA,QAC1B,CAAC;AAGD,kBAAU,QAAQ,MAAM,GAAG,OAAO,aAAa;AAC/C,eAAO;AAAA,MACT;AAAA,MACA,wBAAwB,QAAQ,SAAO,CAAC,GAAG;AACzC,cAAM,iBAAiB;AAAA,UACrB,KAAK,KAAK;AAAA,QACZ;AACA,iBAAS,EAAC,GAAG,gBAAgB,GAAG,OAAM;AAGtC,YAAG,MAAM,QAAQ,MAAM,KAAK,OAAO,WAAW,KAAK,SAAQ;AACzD,eAAK,UAAU,CAAC;AAChB,mBAAQ,IAAI,GAAG,IAAI,OAAO,QAAQ,KAAI;AAIpC,iBAAK,wBAAwB,OAAO,CAAC,GAAG;AAAA,cACtC,KAAK,KAAK,MAAM,OAAO,MAAM,OAAO,MAAM;AAAA,YAC5C,CAAC;AAAA,UACH;AAAA,QACF,OAAK;AACH,gBAAM,YAAY,OAAO,KAAK,KAAK,UAAU;AAC7C,mBAAS,IAAI,GAAG,IAAI,UAAU,QAAQ,KAAK;AACzC,gBAAG,KAAK,cAAc,KAAK,WAAW,UAAU,CAAC,CAAC,CAAC;AAAG;AACtD,kBAAM,MAAM,KAAK,wBAAwB,QAAQ,KAAK,WAAW,UAAU,CAAC,CAAC,EAAE,GAAG;AAClF,gBAAG,KAAK,QAAQ,UAAU,CAAC,CAAC,GAAE;AAC5B,mBAAK,QAAQ,UAAU,CAAC,CAAC,KAAK;AAAA,YAChC,OAAK;AACH,mBAAK,QAAQ,UAAU,CAAC,CAAC,IAAI;AAAA,YAC/B;AAAA,UACF;AAAA,QACF;AAEA,YAAI,UAAU,OAAO,KAAK,KAAK,OAAO,EAAE,IAAI,SAAO;AACjD,iBAAO;AAAA,YACL;AAAA,YACA,YAAY,KAAK,QAAQ,GAAG;AAAA,UAC9B;AAAA,QACF,CAAC;AAED,kBAAU,KAAK,mBAAmB,OAAO;AACzC,kBAAU,QAAQ,MAAM,GAAG,OAAO,GAAG;AAErC,kBAAU,QAAQ,IAAI,UAAQ;AAC5B,iBAAO;AAAA,YACL,MAAM,KAAK,WAAW,KAAK,GAAG,EAAE,KAAK;AAAA,YACrC,YAAY,KAAK;AAAA,YACjB,KAAK,KAAK,WAAW,KAAK,GAAG,EAAE,KAAK,OAAO,KAAK,WAAW,KAAK,GAAG,EAAE,KAAK;AAAA,UAC5E;AAAA,QACF,CAAC;AACD,eAAO;AAAA,MACT;AAAA,MACA,mBAAmB,SAAS;AAC1B,eAAO,QAAQ,KAAK,SAAU,GAAG,GAAG;AAClC,gBAAM,UAAU,EAAE;AAClB,gBAAM,UAAU,EAAE;AAElB,cAAI,UAAU;AACZ,mBAAO;AAET,cAAI,UAAU;AACZ,mBAAO;AAET,iBAAO;AAAA,QACT,CAAC;AAAA,MACH;AAAA;AAAA,MAEA,oBAAoB,OAAO;AACzB,gBAAQ,IAAI,wBAAwB;AACpC,cAAM,OAAO,OAAO,KAAK,KAAK,UAAU;AACxC,YAAI,qBAAqB;AACzB,mBAAW,OAAO,MAAM;AAEtB,gBAAM,OAAO,KAAK,WAAW,GAAG,EAAE,KAAK;AAEvC,cAAG,CAAC,MAAM,KAAK,UAAQ,KAAK,WAAW,KAAK,IAAI,CAAC,GAAG;AAElD,mBAAO,KAAK,WAAW,GAAG;AAC1B;AAEA;AAAA,UACF;AAEA,cAAG,KAAK,QAAQ,GAAG,IAAI,IAAI;AACzB,kBAAM,aAAa,KAAK,WAAW,GAAG,EAAE,KAAK;AAE7C,gBAAG,CAAC,KAAK,WAAW,UAAU,GAAE;AAE9B,qBAAO,KAAK,WAAW,GAAG;AAC1B;AAEA;AAAA,YACF;AAEA,gBAAG,CAAC,KAAK,WAAW,UAAU,EAAE,MAAK;AAEnC,qBAAO,KAAK,WAAW,GAAG;AAC1B;AAEA;AAAA,YACF;AAGA,gBAAG,KAAK,WAAW,UAAU,EAAE,KAAK,YAAa,KAAK,WAAW,UAAU,EAAE,KAAK,SAAS,QAAQ,GAAG,IAAI,GAAI;AAE5G,qBAAO,KAAK,WAAW,GAAG;AAC1B;AAEA;AAAA,YACF;AAAA,UACF;AAAA,QACF;AACA,eAAO,EAAC,oBAAwC,kBAAkB,KAAK,OAAM;AAAA,MAC/E;AAAA,MAEA,IAAI,KAAK;AACP,eAAO,KAAK,WAAW,GAAG,KAAK;AAAA,MACjC;AAAA,MACA,SAAS,KAAK;AACZ,cAAM,YAAY,KAAK,IAAI,GAAG;AAC9B,YAAG,aAAa,UAAU,MAAM;AAC9B,iBAAO,UAAU;AAAA,QACnB;AACA,eAAO;AAAA,MACT;AAAA,MACA,UAAU,KAAK;AACb,cAAM,OAAO,KAAK,SAAS,GAAG;AAC9B,YAAG,QAAQ,KAAK,OAAO;AACrB,iBAAO,KAAK;AAAA,QACd;AACA,eAAO;AAAA,MACT;AAAA,MACA,SAAS,KAAK;AACZ,cAAM,OAAO,KAAK,SAAS,GAAG;AAC9B,YAAG,QAAQ,KAAK,MAAM;AACpB,iBAAO,KAAK;AAAA,QACd;AACA,eAAO;AAAA,MACT;AAAA,MACA,SAAS,KAAK;AACZ,cAAM,OAAO,KAAK,SAAS,GAAG;AAC9B,YAAG,QAAQ,KAAK,MAAM;AACpB,iBAAO,KAAK;AAAA,QACd;AACA,eAAO;AAAA,MACT;AAAA,MACA,aAAa,KAAK;AAChB,cAAM,OAAO,KAAK,SAAS,GAAG;AAC9B,YAAG,QAAQ,KAAK,UAAU;AACxB,iBAAO,KAAK;AAAA,QACd;AACA,eAAO;AAAA,MACT;AAAA,MACA,QAAQ,KAAK;AACX,cAAM,YAAY,KAAK,IAAI,GAAG;AAC9B,YAAG,aAAa,UAAU,KAAK;AAC7B,iBAAO,UAAU;AAAA,QACnB;AACA,eAAO;AAAA,MACT;AAAA,MACA,eAAe,KAAK,KAAK,MAAM;AAC7B,aAAK,WAAW,GAAG,IAAI;AAAA,UACrB;AAAA,UACA;AAAA,QACF;AAAA,MACF;AAAA,MACA,iBAAiB,KAAK,cAAc;AAClC,cAAM,QAAQ,KAAK,UAAU,GAAG;AAChC,YAAG,SAAS,SAAS,cAAc;AACjC,iBAAO;AAAA,QACT;AACA,eAAO;AAAA,MACT;AAAA,MAEA,MAAM,gBAAgB;AACpB,aAAK,aAAa;AAClB,aAAK,aAAa,CAAC;AAEnB,YAAI,mBAAmB,KAAK,MAAM,KAAK,IAAI,IAAI,GAAI;AAEnD,cAAM,KAAK,OAAO,KAAK,WAAW,KAAK,cAAc,iBAAiB,mBAAmB,OAAO;AAEhG,cAAM,KAAK,qBAAqB;AAAA,MAClC;AAAA,IACF;AAEA,IAAAD,QAAO,UAAUC;AAAA;AAAA;;;AC1YjB,IAAM,WAAW,QAAQ,UAAU;AACnC,IAAM,UAAU;AAEhB,IAAM,mBAAmB;AAAA,EACvB,SAAS;AAAA,EACT,WAAW;AAAA,EACX,iBAAiB;AAAA,EACjB,mBAAmB;AAAA,EACnB,mBAAmB;AAAA,EACnB,WAAW;AAAA,EACX,gBAAgB;AAAA,EAChB,eAAe;AAAA,EACf,uBAAuB;AAAA,EACvB,UAAU;AAAA,EACV,YAAY;AAAA,EACZ,kBAAkB;AAAA,EAClB,4BAA4B;AAAA,EAC5B,eAAe;AAAA,EACf,kBAAkB;AAAA,EAClB,WAAW;AAAA,EACX,SAAS;AACX;AACA,IAAM,0BAA0B;AAEhC,IAAI;AACJ,IAAM,uBAAuB,CAAC,MAAM,QAAQ;AAI5C,IAAM,oBAAoB;AAAA,EACxB,MAAM;AAAA,IACJ,WAAW,CAAC,MAAM,KAAK,MAAM,QAAQ,OAAO,QAAQ,MAAM,IAAI;AAAA,IAC9D,UAAU;AAAA,IACV,mBAAmB;AAAA,EACrB;AAAA,EACA,MAAM;AAAA,IACJ,WAAW,CAAC,MAAM,MAAM,SAAM,OAAI;AAAA,IAClC,UAAU;AAAA,IACV,mBAAmB;AAAA,EACrB;AAAA,EACA,MAAM;AAAA,IACJ,WAAW,CAAC,MAAM,OAAO,MAAM,OAAO,OAAO,QAAQ,SAAS,OAAO,MAAM,MAAM,IAAI;AAAA,IACrF,UAAU;AAAA,IACV,mBAAmB;AAAA,EACrB;AAAA,EACA,MAAM;AAAA,IACJ,WAAW,CAAC,QAAQ,SAAS,UAAU,UAAU,UAAU,OAAO,OAAO,SAAS,WAAW,WAAW,SAAS;AAAA,IACjH,UAAU;AAAA,IACV,mBAAmB;AAAA,EACrB;AAAA,EACA,MAAM;AAAA,IACJ,WAAW,CAAC,OAAO,OAAO,QAAQ,OAAO,OAAO,UAAU,UAAU,UAAU,QAAQ;AAAA,IACtF,UAAU;AAAA,IACV,mBAAmB;AAAA,EACrB;AACF;AAGA,IAAM,SAAS,QAAQ,QAAQ;AAE/B,SAAS,IAAI,KAAK;AAChB,SAAO,OAAO,WAAW,KAAK,EAAE,OAAO,GAAG,EAAE,OAAO,KAAK;AAC1D;AAEA,IAAM,yBAAN,cAAqC,SAAS,OAAO;AAAA;AAAA,EAEnD,cAAc;AACZ,UAAM,GAAG,SAAS;AAClB,SAAK,MAAM;AACX,SAAK,oBAAoB;AACzB,SAAK,kBAAkB,CAAC;AACxB,SAAK,UAAU,CAAC;AAChB,SAAK,qBAAqB;AAC1B,SAAK,oBAAoB,CAAC;AAC1B,SAAK,gBAAgB,CAAC;AACtB,SAAK,YAAY,CAAC;AAClB,SAAK,aAAa,CAAC;AACnB,SAAK,WAAW,qBAAqB;AACrC,SAAK,WAAW,kBAAkB,CAAC;AACnC,SAAK,WAAW,oBAAoB,CAAC;AACrC,SAAK,WAAW,QAAQ,CAAC;AACzB,SAAK,WAAW,iBAAiB;AACjC,SAAK,WAAW,oBAAoB,CAAC;AACrC,SAAK,WAAW,cAAc;AAC9B,SAAK,WAAW,wBAAwB;AACxC,SAAK,uBAAuB;AAC5B,SAAK,eAAe;AACpB,SAAK,cAAc,CAAC;AACpB,SAAK,oBAAoB;AACzB,SAAK,mBAAmB;AAAA,EAC1B;AAAA,EAEA,MAAM,SAAS;AAEb,SAAK,IAAI,UAAU,cAAc,KAAK,WAAW,KAAK,IAAI,CAAC;AAAA,EAC7D;AAAA,EACA,WAAW;AACT,SAAK,kBAAkB;AACvB,YAAQ,IAAI,kBAAkB;AAC9B,SAAK,IAAI,UAAU,mBAAmB,2BAA2B;AACjE,SAAK,IAAI,UAAU,mBAAmB,gCAAgC;AAAA,EACxE;AAAA,EACA,MAAM,aAAa;AACjB,YAAQ,IAAI,kCAAkC;AAC9C,cAAU,KAAK,SAAS;AAGxB,UAAM,KAAK,aAAa;AAExB,eAAW,KAAK,iBAAiB,KAAK,IAAI,GAAG,GAAI;AAEjD,gBAAY,KAAK,iBAAiB,KAAK,IAAI,GAAG,KAAQ;AAEtD,SAAK,QAAQ;AACb,SAAK,WAAW;AAAA,MACd,IAAI;AAAA,MACJ,MAAM;AAAA,MACN,MAAM;AAAA,MACN,SAAS,CAAC;AAAA;AAAA,MAEV,gBAAgB,OAAO,WAAW;AAChC,YAAG,OAAO,kBAAkB,GAAG;AAE7B,cAAI,gBAAgB,OAAO,aAAa;AAExC,gBAAM,KAAK,iBAAiB,aAAa;AAAA,QAC3C,OAAO;AAEL,eAAK,gBAAgB,CAAC;AAEtB,gBAAM,KAAK,iBAAiB;AAAA,QAC9B;AAAA,MACF;AAAA,IACF,CAAC;AACD,SAAK,WAAW;AAAA,MACd,IAAI;AAAA,MACJ,MAAM;AAAA,MACN,UAAU,MAAM;AACd,aAAK,UAAU;AAAA,MACjB;AAAA,IACF,CAAC;AAED,SAAK,WAAW;AAAA,MACd,IAAI;AAAA,MACJ,MAAM;AAAA,MACN,UAAU,MAAM;AACd,aAAK,UAAU;AAAA,MACjB;AAAA,IACF,CAAC;AAED,SAAK,WAAW;AAAA,MACd,IAAI;AAAA,MACJ,MAAM;AAAA,MACN,UAAU,MAAM;AACd,aAAK,iBAAiB;AAAA,MACxB;AAAA,IACF,CAAC;AAED,SAAK,cAAc,IAAI,4BAA4B,KAAK,KAAK,IAAI,CAAC;AAElE,SAAK,aAAa,6BAA6B,CAAC,SAAU,IAAI,qBAAqB,MAAM,IAAI,CAAE;AAE/F,SAAK,aAAa,kCAAkC,CAAC,SAAU,IAAI,yBAAyB,MAAM,IAAI,CAAE;AAExG,SAAK,mCAAmC,qBAAqB,KAAK,kBAAkB,KAAK,IAAI,CAAC;AAG9F,QAAG,KAAK,SAAS,WAAW;AAC1B,WAAK,UAAU;AAAA,IACjB;AAEA,QAAG,KAAK,SAAS,WAAW;AAC1B,WAAK,UAAU;AAAA,IACjB;AAEA,QAAG,KAAK,SAAS,YAAY,SAAS;AAEpC,WAAK,SAAS,UAAU;AAExB,YAAM,KAAK,aAAa;AAExB,WAAK,UAAU;AAAA,IACjB;AAEA,SAAK,iBAAiB;AAMtB,SAAK,MAAM,IAAI,YAAY,KAAK,KAAK,IAAI;AAEzC,KAAC,OAAO,gBAAgB,IAAI,KAAK,QAAQ,KAAK,SAAS,MAAM,OAAO,OAAO,gBAAgB,CAAC;AAAA,EAE9F;AAAA,EAEA,MAAM,YAAY;AAChB,SAAK,iBAAiB,IAAI,QAAQ;AAAA,MAChC,aAAa;AAAA,MACb,gBAAgB,KAAK,IAAI,MAAM,QAAQ,OAAO,KAAK,KAAK,IAAI,MAAM,OAAO;AAAA,MACzE,eAAe,KAAK,IAAI,MAAM,QAAQ,MAAM,KAAK,KAAK,IAAI,MAAM,OAAO;AAAA,MACvE,cAAc,KAAK,IAAI,MAAM,QAAQ,KAAK,KAAK,KAAK,IAAI,MAAM,OAAO;AAAA,MACrE,gBAAgB,KAAK,IAAI,MAAM,QAAQ,OAAO,KAAK,KAAK,IAAI,MAAM,OAAO;AAAA,MACzE,cAAc,KAAK,IAAI,MAAM,QAAQ,KAAK,KAAK,KAAK,IAAI,MAAM,OAAO;AAAA,MACrE,eAAe,KAAK,IAAI,MAAM,QAAQ,MAAM,KAAK,KAAK,IAAI,MAAM,OAAO;AAAA,IACzE,CAAC;AACD,SAAK,oBAAoB,MAAM,KAAK,eAAe,KAAK;AACxD,WAAO,KAAK;AAAA,EACd;AAAA,EACA,MAAM,eAAe;AAEnB,QAAG,CAAC,KAAK,SAAS;AAAa,aAAO,IAAI,SAAS,OAAO,2EAA2E;AAErI,UAAM,KAAK,OAAO,GAAG,SAAS,YAAY;AAAA,MACxC,KAAK;AAAA,MACL,QAAQ;AAAA,MACR,SAAS;AAAA,QACP,gBAAgB;AAAA,MAClB;AAAA,MACA,MAAM,KAAK,UAAU;AAAA,QACnB,aAAa,KAAK,SAAS;AAAA,MAC7B,CAAC;AAAA,IACH,CAAC;AACD,QAAG,GAAG,WAAW;AAAK,aAAO,QAAQ,MAAM,+BAA+B,EAAE;AAC5E,YAAQ,IAAI,EAAE;AACd,UAAM,KAAK,IAAI,MAAM,QAAQ,MAAM,+CAA+C,GAAG,KAAK,IAAI;AAC9F,UAAM,KAAK,IAAI,MAAM,QAAQ,MAAM,qDAAqD,GAAG,KAAK,QAAQ;AACxG,UAAM,KAAK,IAAI,MAAM,QAAQ,MAAM,kDAAkD,GAAG,KAAK,MAAM;AACnG,WAAO,iBAAiB,OAAO,OAAO;AACpC,cAAQ,IAAI,qBAAqB,EAAE;AACnC,YAAM,OAAO,IAAI,QAAQ,cAAc,EAAE;AACzC,YAAM,OAAO,IAAI,QAAQ,aAAa,EAAE;AACxC,cAAQ,IAAI,oBAAoB,EAAE;AAAA,IACpC;AACA,WAAO,eAAe,KAAK,SAAS,EAAE;AAAA,EACxC;AAAA,EAEA,MAAM,eAAe;AACnB,SAAK,WAAW,OAAO,OAAO,CAAC,GAAG,kBAAkB,MAAM,KAAK,SAAS,CAAC;AAEzE,QAAG,KAAK,SAAS,mBAAmB,KAAK,SAAS,gBAAgB,SAAS,GAAG;AAE5E,WAAK,kBAAkB,KAAK,SAAS,gBAAgB,MAAM,GAAG,EAAE,IAAI,CAAC,SAAS;AAC5E,eAAO,KAAK,KAAK;AAAA,MACnB,CAAC;AAAA,IACH;AAEA,QAAG,KAAK,SAAS,qBAAqB,KAAK,SAAS,kBAAkB,SAAS,GAAG;AAEhF,YAAM,oBAAoB,KAAK,SAAS,kBAAkB,MAAM,GAAG,EAAE,IAAI,CAAC,WAAW;AAEnF,iBAAS,OAAO,KAAK;AACrB,YAAG,OAAO,MAAM,EAAE,MAAM,KAAK;AAC3B,iBAAO,SAAS;AAAA,QAClB,OAAO;AACL,iBAAO;AAAA,QACT;AAAA,MACF,CAAC;AAED,WAAK,kBAAkB,KAAK,gBAAgB,OAAO,iBAAiB;AAAA,IACtE;AAEA,QAAG,KAAK,SAAS,qBAAqB,KAAK,SAAS,kBAAkB,SAAS,GAAG;AAChF,WAAK,oBAAoB,KAAK,SAAS,kBAAkB,MAAM,GAAG,EAAE,IAAI,CAAC,WAAW;AAClF,eAAO,OAAO,KAAK;AAAA,MACrB,CAAC;AAAA,IACH;AAEA,QAAG,KAAK,SAAS,aAAa,KAAK,SAAS,UAAU,SAAS,GAAG;AAChE,WAAK,YAAY,KAAK,SAAS,UAAU,MAAM,GAAG,EAAE,IAAI,CAAC,SAAS;AAChE,eAAO,KAAK,KAAK;AAAA,MACnB,CAAC;AAAA,IACH;AAEA,SAAK,oBAAoB,IAAI,OAAO,OAAO,kBAAkB,KAAK,SAAS,QAAQ,EAAE,QAAQ,KAAK,GAAG,SAAS,IAAI;AAElH,UAAM,KAAK,kBAAkB;AAAA,EAC/B;AAAA,EACA,MAAM,aAAa,WAAS,OAAO;AACjC,UAAM,KAAK,SAAS,KAAK,QAAQ;AAEjC,UAAM,KAAK,aAAa;AAExB,QAAG,UAAU;AACX,WAAK,gBAAgB,CAAC;AACtB,YAAM,KAAK,iBAAiB;AAAA,IAC9B;AAAA,EACF;AAAA;AAAA,EAGA,MAAM,mBAAmB;AAEvB,QAAI;AAEF,YAAM,WAAW,OAAO,GAAG,SAAS,YAAY;AAAA,QAC9C,KAAK;AAAA,QACL,QAAQ;AAAA,QACR,SAAS;AAAA,UACP,gBAAgB;AAAA,QAClB;AAAA,QACA,aAAa;AAAA,MACf,CAAC;AAED,YAAM,iBAAiB,KAAK,MAAM,SAAS,IAAI,EAAE;AAGjD,UAAG,mBAAmB,SAAS;AAC7B,YAAI,SAAS,OAAO,qDAAqD,iBAAiB;AAC1F,aAAK,mBAAmB;AACxB,aAAK,aAAa,KAAK;AAAA,MACzB;AAAA,IACF,SAAS,OAAP;AACA,cAAQ,IAAI,KAAK;AAAA,IACnB;AAAA,EACF;AAAA,EAEA,MAAM,kBAAkB,UAAU,WAAW,KAAK;AAChD,QAAI;AACJ,QAAG,SAAS,KAAK,EAAE,SAAS,GAAG;AAC7B,gBAAU,MAAM,KAAK,IAAI,OAAO,QAAQ;AAAA,IAC1C,OAAO;AAEL,cAAQ,IAAI,GAAG;AACf,YAAM,OAAO,KAAK,IAAI,MAAM,sBAAsB,IAAI,UAAU;AAChE,gBAAU,MAAM,KAAK,sBAAsB,IAAI;AAAA,IACjD;AACA,QAAI,QAAQ,QAAQ;AAClB,WAAK,eAAe,WAAW,OAAO;AAAA,IACxC;AAAA,EACF;AAAA,EAEA,MAAM,iBAAiB,gBAAc,MAAM;AACzC,QAAI,OAAO,KAAK,SAAS;AACzB,QAAI,CAAC,MAAM;AAET,YAAM,KAAK,UAAU;AACrB,aAAO,KAAK,SAAS;AAAA,IACvB;AACA,UAAM,KAAK,mBAAmB,aAAa;AAAA,EAC7C;AAAA,EAEA,UAAS;AACP,aAAS,QAAQ,qBAAqB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,wDAMc;AAAA,EACtD;AAAA;AAAA,EAGA,MAAM,mBAAmB;AACvB,UAAM,YAAY,KAAK,IAAI,UAAU,cAAc;AACnD,UAAM,WAAW,IAAI,UAAU,IAAI;AAEnC,QAAG,OAAO,KAAK,cAAc,QAAQ,MAAM,aAAa;AACtD,UAAI,SAAS,OAAO,uFAAuF;AAC3G;AAAA,IACF;AAEA,UAAM,OAAO,KAAK,MAAM,KAAK,OAAO,IAAI,KAAK,cAAc,QAAQ,EAAE,SAAO,CAAC;AAC7E,UAAM,cAAc,KAAK,cAAc,QAAQ,EAAE,IAAI;AAErD,SAAK,UAAU,WAAW;AAAA,EAC5B;AAAA,EAEA,MAAM,YAAY;AAChB,QAAG,KAAK,SAAS,GAAE;AACjB,cAAQ,IAAI,qCAAqC;AACjD;AAAA,IACF;AACA,SAAK,IAAI,UAAU,mBAAmB,2BAA2B;AACjE,UAAM,KAAK,IAAI,UAAU,aAAa,KAAK,EAAE,aAAa;AAAA,MACxD,MAAM;AAAA,MACN,QAAQ;AAAA,IACV,CAAC;AACD,SAAK,IAAI,UAAU;AAAA,MACjB,KAAK,IAAI,UAAU,gBAAgB,2BAA2B,EAAE,CAAC;AAAA,IACnE;AAAA,EACF;AAAA;AAAA,EAEA,WAAW;AACT,aAAS,QAAQ,KAAK,IAAI,UAAU,gBAAgB,2BAA2B,GAAG;AAChF,UAAI,KAAK,gBAAgB,sBAAsB;AAC7C,eAAO,KAAK;AAAA,MACd;AAAA,IACF;AAAA,EACF;AAAA;AAAA,EAEA,MAAM,UAAU,UAAQ,GAAG;AACzB,QAAG,CAAC,KAAK,mBAAmB;AAC1B,cAAQ,IAAI,2BAA2B;AACvC,UAAG,UAAU,GAAG;AAEd,mBAAW,MAAM;AACf,eAAK,UAAU,UAAQ,CAAC;AAAA,QAC1B,GAAG,OAAQ,UAAQ,EAAE;AACrB;AAAA,MACF;AACA,cAAQ,IAAI,iDAAiD;AAC7D,WAAK,UAAU;AACf;AAAA,IACF;AACA,SAAK,IAAI,UAAU,mBAAmB,gCAAgC;AACtE,UAAM,KAAK,IAAI,UAAU,aAAa,KAAK,EAAE,aAAa;AAAA,MACxD,MAAM;AAAA,MACN,QAAQ;AAAA,IACV,CAAC;AACD,SAAK,IAAI,UAAU;AAAA,MACjB,KAAK,IAAI,UAAU,gBAAgB,gCAAgC,EAAE,CAAC;AAAA,IACxE;AAAA,EACF;AAAA;AAAA,EAGA,MAAM,qBAAqB;AAEzB,UAAM,SAAS,MAAM,KAAK,IAAI,MAAM,SAAS,GAAG,OAAO,CAAC,SAAS,gBAAgB,SAAS,UAAU,KAAK,cAAc,QAAQ,KAAK,cAAc,SAAS;AAG3J,UAAM,aAAa,KAAK,IAAI,UAAU,gBAAgB,UAAU,EAAE,IAAI,CAAC,SAAS,KAAK,KAAK,IAAI;AAC9F,UAAM,eAAe,KAAK,eAAe,oBAAoB,KAAK;AAClE,QAAG,KAAK,SAAS,YAAW;AAC1B,WAAK,WAAW,cAAc,MAAM;AACpC,WAAK,WAAW,qBAAqB,aAAa;AAClD,WAAK,WAAW,mBAAmB,aAAa;AAAA,IAClD;AAEA,QAAI,iBAAiB,CAAC;AACtB,aAAS,IAAI,GAAG,IAAI,MAAM,QAAQ,KAAK;AAErC,UAAG,MAAM,CAAC,EAAE,KAAK,QAAQ,GAAG,IAAI,IAAI;AAElC,aAAK,cAAc,iBAAiB;AACpC;AAAA,MACF;AAEA,UAAG,KAAK,eAAe,iBAAiB,IAAI,MAAM,CAAC,EAAE,IAAI,GAAG,MAAM,CAAC,EAAE,KAAK,KAAK,GAAG;AAGhF;AAAA,MACF;AAEA,UAAG,KAAK,SAAS,aAAa,QAAQ,MAAM,CAAC,EAAE,IAAI,IAAI,IAAI;AAIzD,YAAG,KAAK,sBAAsB;AAC5B,uBAAa,KAAK,oBAAoB;AACtC,eAAK,uBAAuB;AAAA,QAC9B;AAEA,YAAG,CAAC,KAAK,4BAA2B;AAClC,cAAI,SAAS,OAAO,qFAAqF;AACzG,eAAK,6BAA6B;AAClC,qBAAW,MAAM;AACf,iBAAK,6BAA6B;AAAA,UACpC,GAAG,GAAM;AAAA,QACX;AACA;AAAA,MACF;AAEA,UAAI,OAAO;AACX,eAAQ,IAAI,GAAG,IAAI,KAAK,gBAAgB,QAAQ,KAAK;AACnD,YAAG,MAAM,CAAC,EAAE,KAAK,QAAQ,KAAK,gBAAgB,CAAC,CAAC,IAAI,IAAI;AACtD,iBAAO;AACP,eAAK,cAAc,KAAK,gBAAgB,CAAC,CAAC;AAE1C;AAAA,QACF;AAAA,MACF;AACA,UAAG,MAAM;AACP;AAAA,MACF;AAEA,UAAG,WAAW,QAAQ,MAAM,CAAC,CAAC,IAAI,IAAI;AAEpC;AAAA,MACF;AACA,UAAI;AAEF,uBAAe,KAAK,KAAK,oBAAoB,MAAM,CAAC,GAAG,KAAK,CAAC;AAAA,MAC/D,SAAS,OAAP;AACA,gBAAQ,IAAI,KAAK;AAAA,MACnB;AAEA,UAAG,eAAe,SAAS,GAAG;AAE5B,cAAM,QAAQ,IAAI,cAAc;AAEhC,yBAAiB,CAAC;AAAA,MACpB;AAGA,UAAG,IAAI,KAAK,IAAI,QAAQ,GAAG;AACzB,cAAM,KAAK,wBAAwB;AAAA,MACrC;AAAA,IACF;AAEA,UAAM,QAAQ,IAAI,cAAc;AAEhC,UAAM,KAAK,wBAAwB;AAEnC,QAAG,KAAK,WAAW,kBAAkB,SAAS,GAAG;AAC/C,YAAM,KAAK,uBAAuB;AAAA,IACpC;AAAA,EACF;AAAA,EAEA,MAAM,wBAAwB,QAAM,OAAO;AACzC,QAAG,CAAC,KAAK,oBAAmB;AAC1B;AAAA,IACF;AAEA,QAAG,CAAC,OAAO;AAET,UAAG,KAAK,cAAc;AACpB,qBAAa,KAAK,YAAY;AAC9B,aAAK,eAAe;AAAA,MACtB;AACA,WAAK,eAAe,WAAW,MAAM;AAEnC,aAAK,wBAAwB,IAAI;AAEjC,YAAG,KAAK,cAAc;AACpB,uBAAa,KAAK,YAAY;AAC9B,eAAK,eAAe;AAAA,QACtB;AAAA,MACF,GAAG,GAAK;AACR,cAAQ,IAAI,gBAAgB;AAC5B;AAAA,IACF;AAEA,QAAG;AAED,YAAM,KAAK,eAAe,KAAK;AAC/B,WAAK,qBAAqB;AAAA,IAC5B,SAAO,OAAN;AACC,cAAQ,IAAI,KAAK;AACjB,UAAI,SAAS,OAAO,wBAAsB,MAAM,OAAO;AAAA,IACzD;AAAA,EAEF;AAAA;AAAA,EAEA,MAAM,yBAA0B;AAE9B,QAAI,oBAAoB,CAAC;AAEzB,UAAM,gCAAgC,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,0CAA0C;AACpH,QAAG,+BAA+B;AAChC,0BAAoB,MAAM,KAAK,IAAI,MAAM,QAAQ,KAAK,0CAA0C;AAEhG,0BAAoB,kBAAkB,MAAM,MAAM;AAAA,IACpD;AAEA,wBAAoB,kBAAkB,OAAO,KAAK,WAAW,iBAAiB;AAE9E,wBAAoB,CAAC,GAAG,IAAI,IAAI,iBAAiB,CAAC;AAElD,sBAAkB,KAAK;AAEvB,wBAAoB,kBAAkB,KAAK,MAAM;AAEjD,UAAM,KAAK,IAAI,MAAM,QAAQ,MAAM,4CAA4C,iBAAiB;AAEhG,UAAM,KAAK,kBAAkB;AAAA,EAC/B;AAAA;AAAA,EAGA,MAAM,oBAAqB;AAEzB,UAAM,gCAAgC,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,0CAA0C;AACpH,QAAG,CAAC,+BAA+B;AACjC,WAAK,SAAS,eAAe,CAAC;AAC9B,cAAQ,IAAI,kBAAkB;AAC9B;AAAA,IACF;AAEA,UAAM,oBAAoB,MAAM,KAAK,IAAI,MAAM,QAAQ,KAAK,0CAA0C;AAEtG,UAAM,0BAA0B,kBAAkB,MAAM,MAAM;AAE9D,UAAM,eAAe,wBAAwB,IAAI,eAAa,UAAU,MAAM,GAAG,EAAE,CAAC,CAAC,EAAE,OAAO,CAAC,QAAQ,SAAS,OAAO,SAAS,IAAI,IAAI,SAAS,CAAC,GAAG,QAAQ,IAAI,GAAG,CAAC,CAAC;AAEtK,SAAK,SAAS,eAAe;AAAA,EAE/B;AAAA;AAAA,EAEA,MAAM,qBAAsB;AAE1B,SAAK,SAAS,eAAe,CAAC;AAE9B,UAAM,gCAAgC,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,0CAA0C;AACpH,QAAG,+BAA+B;AAChC,YAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,0CAA0C;AAAA,IAChF;AAEA,UAAM,KAAK,mBAAmB;AAAA,EAChC;AAAA;AAAA,EAIA,MAAM,mBAAmB;AACvB,QAAG,CAAE,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,YAAY,GAAI;AACvD;AAAA,IACF;AACA,QAAI,iBAAiB,MAAM,KAAK,IAAI,MAAM,QAAQ,KAAK,YAAY;AAEnE,QAAI,eAAe,QAAQ,oBAAoB,IAAI,GAAG;AAEpD,UAAI,mBAAmB;AACvB,0BAAoB;AACpB,YAAM,KAAK,IAAI,MAAM,QAAQ,MAAM,cAAc,iBAAiB,gBAAgB;AAClF,cAAQ,IAAI,wCAAwC;AAAA,IACtD;AAAA,EACF;AAAA;AAAA,EAGA,MAAM,gCAAgC;AACpC,QAAI,SAAS,OAAO,+EAA+E;AAEnG,UAAM,KAAK,eAAe,cAAc;AAExC,UAAM,KAAK,mBAAmB;AAC9B,SAAK,kBAAkB;AACvB,QAAI,SAAS,OAAO,2EAA2E;AAAA,EACjG;AAAA;AAAA,EAGA,MAAM,oBAAoB,WAAW,OAAK,MAAM;AAE9C,QAAI,YAAY,CAAC;AACjB,QAAI,SAAS,CAAC;AAEd,UAAM,gBAAgB,IAAI,UAAU,IAAI;AAExC,QAAI,mBAAmB,UAAU,KAAK,QAAQ,OAAO,EAAE;AACvD,uBAAmB,iBAAiB,QAAQ,OAAO,KAAK;AAExD,QAAI,YAAY;AAChB,aAAQ,IAAI,GAAG,IAAI,KAAK,UAAU,QAAQ,KAAK;AAC7C,UAAG,UAAU,KAAK,QAAQ,KAAK,UAAU,CAAC,CAAC,IAAI,IAAI;AACjD,oBAAY;AACZ,gBAAQ,IAAI,mCAAmC,KAAK,UAAU,CAAC,CAAC;AAEhE;AAAA,MACF;AAAA,IACF;AAEA,QAAG,WAAW;AACZ,gBAAU,KAAK,CAAC,eAAe,kBAAkB;AAAA,QAC/C,OAAO,UAAU,KAAK;AAAA,QACtB,MAAM,UAAU;AAAA,MAClB,CAAC,CAAC;AACF,YAAM,KAAK,qBAAqB,SAAS;AACzC;AAAA,IACF;AAIA,QAAG,UAAU,cAAc,UAAU;AAEnC,YAAM,kBAAkB,MAAM,KAAK,IAAI,MAAM,WAAW,SAAS;AACjE,UAAI,OAAO,oBAAoB,YAAc,gBAAgB,QAAQ,OAAO,IAAI,IAAK;AACnF,cAAM,cAAc,KAAK,MAAM,eAAe;AAE9C,iBAAQ,IAAI,GAAG,IAAI,YAAY,MAAM,QAAQ,KAAK;AAEhD,cAAG,YAAY,MAAM,CAAC,EAAE,MAAM;AAE5B,gCAAoB,OAAO,YAAY,MAAM,CAAC,EAAE;AAAA,UAClD;AAEA,cAAG,YAAY,MAAM,CAAC,EAAE,MAAM;AAE5B,gCAAoB,aAAa,YAAY,MAAM,CAAC,EAAE;AAAA,UACxD;AAAA,QACF;AAAA,MACF;AAEA,gBAAU,KAAK,CAAC,eAAe,kBAAkB;AAAA,QAC/C,OAAO,UAAU,KAAK;AAAA,QACtB,MAAM,UAAU;AAAA,MAClB,CAAC,CAAC;AACF,YAAM,KAAK,qBAAqB,SAAS;AACzC;AAAA,IACF;AAMA,UAAM,gBAAgB,MAAM,KAAK,IAAI,MAAM,WAAW,SAAS;AAC/D,QAAI,4BAA4B;AAChC,UAAM,gBAAgB,KAAK,aAAa,eAAe,UAAU,IAAI;AAGrE,QAAG,cAAc,SAAS,GAAG;AAG3B,eAAS,IAAI,GAAG,IAAI,cAAc,QAAQ,KAAK;AAE7C,cAAM,oBAAoB,cAAc,CAAC,EAAE;AAG3C,cAAM,YAAY,IAAI,cAAc,CAAC,EAAE,IAAI;AAC3C,eAAO,KAAK,SAAS;AAGrB,YAAI,KAAK,eAAe,SAAS,SAAS,MAAM,kBAAkB,QAAQ;AAGxE;AAAA,QACF;AAGA,YAAG,KAAK,eAAe,iBAAiB,WAAW,UAAU,KAAK,KAAK,GAAG;AAGxE;AAAA,QACF;AAEA,cAAM,aAAa,IAAI,kBAAkB,KAAK,CAAC;AAC/C,YAAG,KAAK,eAAe,SAAS,SAAS,MAAM,YAAY;AAGzD;AAAA,QACF;AAGA,kBAAU,KAAK,CAAC,WAAW,mBAAmB;AAAA;AAAA;AAAA,UAG5C,OAAO,KAAK,IAAI;AAAA,UAChB,MAAM;AAAA,UACN,QAAQ;AAAA,UACR,MAAM,cAAc,CAAC,EAAE;AAAA,UACvB,MAAM,kBAAkB;AAAA,QAC1B,CAAC,CAAC;AACF,YAAG,UAAU,SAAS,GAAG;AAEvB,gBAAM,KAAK,qBAAqB,SAAS;AACzC,uCAA6B,UAAU;AAGvC,cAAI,6BAA6B,IAAI;AAEnC,kBAAM,KAAK,wBAAwB;AAEnC,wCAA4B;AAAA,UAC9B;AAEA,sBAAY,CAAC;AAAA,QACf;AAAA,MACF;AAAA,IACF;AAEA,QAAG,UAAU,SAAS,GAAG;AAEvB,YAAM,KAAK,qBAAqB,SAAS;AACzC,kBAAY,CAAC;AACb,mCAA6B,UAAU;AAAA,IACzC;AAQA,wBAAoB;AAAA;AAIpB,QAAG,cAAc,SAAS,yBAAyB;AACjD,0BAAoB;AAAA,IACtB,OAAK;AACH,YAAM,kBAAkB,KAAK,IAAI,cAAc,aAAa,SAAS;AAErE,UAAG,OAAO,gBAAgB,aAAa,aAAa;AAElD,4BAAoB,cAAc,UAAU,GAAG,uBAAuB;AAAA,MACxE,OAAK;AACH,YAAI,gBAAgB;AACpB,iBAAS,IAAI,GAAG,IAAI,gBAAgB,SAAS,QAAQ,KAAK;AAExD,gBAAM,gBAAgB,gBAAgB,SAAS,CAAC,EAAE;AAElD,gBAAM,eAAe,gBAAgB,SAAS,CAAC,EAAE;AAEjD,cAAI,aAAa;AACjB,mBAAS,IAAI,GAAG,IAAI,eAAe,KAAK;AACtC,0BAAc;AAAA,UAChB;AAEA,2BAAiB,GAAG,cAAc;AAAA;AAAA,QACpC;AAEA,4BAAoB;AACpB,YAAG,iBAAiB,SAAS,yBAAyB;AACpD,6BAAmB,iBAAiB,UAAU,GAAG,uBAAuB;AAAA,QAC1E;AAAA,MACF;AAAA,IACF;AAGA,UAAM,YAAY,IAAI,iBAAiB,KAAK,CAAC;AAC7C,UAAM,gBAAgB,KAAK,eAAe,SAAS,aAAa;AAChE,QAAG,iBAAkB,cAAc,eAAgB;AAEjD,WAAK,kBAAkB,QAAQ,gBAAgB;AAC/C;AAAA,IACF;AAAC;AAGD,UAAM,kBAAkB,KAAK,eAAe,aAAa,aAAa;AACtE,QAAI,0BAA0B;AAC9B,QAAG,mBAAmB,MAAM,QAAQ,eAAe,KAAM,OAAO,SAAS,GAAI;AAE3E,eAAS,IAAI,GAAG,IAAI,OAAO,QAAQ,KAAK;AACtC,YAAG,gBAAgB,QAAQ,OAAO,CAAC,CAAC,MAAM,IAAI;AAC5C,oCAA0B;AAC1B;AAAA,QACF;AAAA,MACF;AAAA,IACF;AAEA,QAAG,yBAAwB;AAEzB,YAAM,iBAAiB,UAAU,KAAK;AAEtC,YAAM,iBAAiB,KAAK,eAAe,SAAS,aAAa;AACjE,UAAI,gBAAgB;AAElB,cAAM,iBAAiB,KAAK,MAAO,KAAK,IAAI,iBAAiB,cAAc,IAAI,iBAAkB,GAAG;AACpG,YAAG,iBAAiB,IAAI;AAGtB,eAAK,WAAW,kBAAkB,UAAU,IAAI,IAAI,iBAAiB;AACrE,eAAK,kBAAkB,QAAQ,gBAAgB;AAC/C;AAAA,QACF;AAAA,MACF;AAAA,IACF;AACA,QAAI,OAAO;AAAA,MACT,OAAO,UAAU,KAAK;AAAA,MACtB,MAAM;AAAA,MACN,MAAM,UAAU;AAAA,MAChB,MAAM,UAAU,KAAK;AAAA,MACrB,UAAU;AAAA,IACZ;AAEA,cAAU,KAAK,CAAC,eAAe,kBAAkB,IAAI,CAAC;AAEtD,UAAM,KAAK,qBAAqB,SAAS;AAIzC,QAAI,MAAM;AAER,YAAM,KAAK,wBAAwB;AAAA,IACrC;AAAA,EAEF;AAAA,EAEA,kBAAkB,QAAQ,kBAAkB;AAC1C,QAAI,OAAO,SAAS,GAAG;AAErB,WAAK,WAAW,yBAAyB,iBAAiB,SAAS;AAAA,IACrE,OAAO;AAEL,WAAK,WAAW,yBAAyB,iBAAiB,SAAS;AAAA,IACrE;AAAA,EACF;AAAA,EAEA,MAAM,qBAAqB,WAAW;AACpC,YAAQ,IAAI,sBAAsB;AAElC,QAAG,UAAU,WAAW;AAAG;AAE3B,UAAM,eAAe,UAAU,IAAI,CAAC,QAAQ,IAAI,CAAC,CAAC;AAElD,UAAM,iBAAiB,MAAM,KAAK,6BAA6B,YAAY;AAE3E,QAAG,CAAC,gBAAgB;AAClB,cAAQ,IAAI,wBAAwB;AAEpC,WAAK,WAAW,oBAAoB,CAAC,GAAG,KAAK,WAAW,mBAAmB,GAAG,UAAU,IAAI,CAAC,QAAQ,IAAI,CAAC,EAAE,IAAI,CAAC;AACjH;AAAA,IACF;AAEA,QAAG,gBAAe;AAChB,WAAK,qBAAqB;AAE1B,UAAG,KAAK,SAAS,YAAW;AAC1B,YAAG,KAAK,SAAS,kBAAiB;AAChC,eAAK,WAAW,QAAQ,CAAC,GAAG,KAAK,WAAW,OAAO,GAAG,UAAU,IAAI,CAAC,QAAQ,IAAI,CAAC,EAAE,IAAI,CAAC;AAAA,QAC3F;AACA,aAAK,WAAW,kBAAkB,UAAU;AAE5C,aAAK,WAAW,eAAe,eAAe,MAAM;AAAA,MACtD;AAGA,eAAQ,IAAI,GAAG,IAAI,eAAe,KAAK,QAAQ,KAAK;AAClD,cAAM,MAAM,eAAe,KAAK,CAAC,EAAE;AACnC,cAAM,QAAQ,eAAe,KAAK,CAAC,EAAE;AACrC,YAAG,KAAK;AACN,gBAAM,MAAM,UAAU,KAAK,EAAE,CAAC;AAC9B,gBAAM,OAAO,UAAU,KAAK,EAAE,CAAC;AAC/B,eAAK,eAAe,eAAe,KAAK,KAAK,IAAI;AAAA,QACnD;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAAA,EAEA,MAAM,6BAA6B,aAAa,UAAU,GAAG;AAS3D,QAAG,YAAY,WAAW,GAAG;AAC3B,cAAQ,IAAI,sBAAsB;AAClC,aAAO;AAAA,IACT;AACA,UAAM,aAAa;AAAA,MACjB,OAAO;AAAA,MACP,OAAO;AAAA,IACT;AAEA,UAAM,YAAY;AAAA,MAChB,KAAK;AAAA,MACL,QAAQ;AAAA,MACR,MAAM,KAAK,UAAU,UAAU;AAAA,MAC/B,SAAS;AAAA,QACP,gBAAgB;AAAA,QAChB,iBAAiB,UAAU,KAAK,SAAS;AAAA,MAC3C;AAAA,IACF;AACA,QAAI;AACJ,QAAI;AACF,aAAO,OAAO,GAAG,SAAS,SAAS,SAAS;AAC5C,aAAO,KAAK,MAAM,IAAI;AAAA,IACxB,SAAS,OAAP;AAEA,UAAI,MAAM,WAAW,OAAS,UAAU,GAAI;AAC1C;AAEA,cAAM,UAAU,KAAK,IAAI,SAAS,CAAC;AACnC,gBAAQ,IAAI,6BAA6B,oBAAoB;AAC7D,cAAM,IAAI,QAAQ,OAAK,WAAW,GAAG,MAAO,OAAO,CAAC;AACpD,eAAO,MAAM,KAAK,6BAA6B,aAAa,OAAO;AAAA,MACrE;AAEA,cAAQ,IAAI,IAAI;AAOhB,cAAQ,IAAI,KAAK;AAGjB,aAAO;AAAA,IACT;AAAA,EACF;AAAA,EACA,MAAM,eAAe;AACnB,UAAM,cAAc;AACpB,UAAM,OAAO,MAAM,KAAK,6BAA6B,WAAW;AAChE,QAAG,QAAQ,KAAK,OAAO;AACrB,cAAQ,IAAI,kBAAkB;AAC9B,aAAO;AAAA,IACT,OAAK;AACH,cAAQ,IAAI,oBAAoB;AAChC,aAAO;AAAA,IACT;AAAA,EACF;AAAA,EAGA,oBAAoB;AAElB,QAAG,KAAK,SAAS,YAAY;AAC3B,UAAI,KAAK,WAAW,mBAAmB,GAAG;AACxC;AAAA,MACF,OAAK;AAEH,gBAAQ,IAAI,KAAK,UAAU,KAAK,YAAY,MAAM,CAAC,CAAC;AAAA,MACtD;AAAA,IACF;AAGA,SAAK,aAAa,CAAC;AACnB,SAAK,WAAW,qBAAqB;AACrC,SAAK,WAAW,kBAAkB,CAAC;AACnC,SAAK,WAAW,oBAAoB,CAAC;AACrC,SAAK,WAAW,QAAQ,CAAC;AACzB,SAAK,WAAW,iBAAiB;AACjC,SAAK,WAAW,oBAAoB,CAAC;AACrC,SAAK,WAAW,cAAc;AAC9B,SAAK,WAAW,wBAAwB;AAAA,EAC1C;AAAA;AAAA,EAGA,MAAM,sBAAsB,eAAa,MAAM;AAE7C,UAAM,WAAW,IAAI,aAAa,IAAI;AAGtC,QAAI,UAAU,CAAC;AACf,QAAG,KAAK,cAAc,QAAQ,GAAG;AAC/B,gBAAU,KAAK,cAAc,QAAQ;AAAA,IAEvC,OAAK;AAEH,eAAQ,IAAI,GAAG,IAAI,KAAK,gBAAgB,QAAQ,KAAK;AACnD,YAAG,aAAa,KAAK,QAAQ,KAAK,gBAAgB,CAAC,CAAC,IAAI,IAAI;AAC1D,eAAK,cAAc,KAAK,gBAAgB,CAAC,CAAC;AAE1C,iBAAO;AAAA,QACT;AAAA,MACF;AAIA,iBAAW,MAAM;AACf,aAAK,mBAAmB;AAAA,MAC1B,GAAG,GAAI;AAEP,UAAG,KAAK,eAAe,iBAAiB,UAAU,aAAa,KAAK,KAAK,GAAG;AAAA,MAG5E,OAAK;AAEH,cAAM,KAAK,oBAAoB,YAAY;AAAA,MAC7C;AAEA,YAAM,MAAM,KAAK,eAAe,QAAQ,QAAQ;AAChD,UAAG,CAAC,KAAK;AACP,eAAO,mCAAiC,aAAa;AAAA,MACvD;AAGA,gBAAU,KAAK,eAAe,QAAQ,KAAK;AAAA,QACzC,UAAU;AAAA,QACV,eAAe,KAAK,SAAS;AAAA,MAC/B,CAAC;AAGD,WAAK,cAAc,QAAQ,IAAI;AAAA,IACjC;AAGA,WAAO;AAAA,EACT;AAAA;AAAA,EAGA,cAAc,WAAW;AAEvB,SAAK,WAAW,gBAAgB,SAAS,KAAK,KAAK,WAAW,gBAAgB,SAAS,KAAK,KAAK;AAAA,EACnG;AAAA,EAGA,aAAa,UAAU,WAAU;AAE/B,QAAG,KAAK,SAAS,eAAe;AAC9B,aAAO,CAAC;AAAA,IACV;AAEA,UAAM,QAAQ,SAAS,MAAM,IAAI;AAEjC,QAAI,SAAS,CAAC;AAEd,QAAI,iBAAiB,CAAC;AAEtB,UAAM,mBAAmB,UAAU,QAAQ,OAAO,EAAE,EAAE,QAAQ,OAAO,KAAK;AAE1E,QAAI,QAAQ;AACZ,QAAI,iBAAiB;AACrB,QAAI,aAAa;AAEjB,QAAI,oBAAoB;AACxB,QAAI,IAAI;AACR,QAAI,sBAAsB,CAAC;AAE3B,SAAK,IAAI,GAAG,IAAI,MAAM,QAAQ,KAAK;AAEjC,YAAM,OAAO,MAAM,CAAC;AAIpB,UAAI,CAAC,KAAK,WAAW,GAAG,KAAM,CAAC,KAAI,GAAG,EAAE,QAAQ,KAAK,CAAC,CAAC,IAAI,GAAG;AAE5D,YAAG,SAAS;AAAI;AAEhB,YAAG,CAAC,MAAM,QAAQ,EAAE,QAAQ,IAAI,IAAI;AAAI;AAExC,YAAG,eAAe,WAAW;AAAG;AAEhC,iBAAS,OAAO;AAChB;AAAA,MACF;AAKA,0BAAoB;AAEpB,UAAG,IAAI,KAAM,sBAAuB,IAAE,KAAQ,MAAM,QAAQ,IAAI,IAAI,MAAO,KAAK,kBAAkB,cAAc,GAAG;AACjH,qBAAa;AAAA,MACf;AAEA,YAAM,QAAQ,KAAK,MAAM,GAAG,EAAE,SAAS;AAEvC,uBAAiB,eAAe,OAAO,YAAU,OAAO,QAAQ,KAAK;AAGrE,qBAAe,KAAK,EAAC,QAAQ,KAAK,QAAQ,MAAM,EAAE,EAAE,KAAK,GAAG,MAAY,CAAC;AAEzE,cAAQ;AACR,eAAS,OAAO,eAAe,IAAI,YAAU,OAAO,MAAM,EAAE,KAAK,KAAK;AACtE,uBAAiB,MAAI,eAAe,IAAI,YAAU,OAAO,MAAM,EAAE,KAAK,GAAG;AAEzE,UAAG,oBAAoB,QAAQ,cAAc,IAAI,IAAI;AACnD,YAAI,QAAQ;AACZ,eAAM,oBAAoB,QAAQ,GAAG,kBAAkB,QAAQ,IAAI,IAAI;AACrE;AAAA,QACF;AACA,yBAAiB,GAAG,kBAAkB;AAAA,MACxC;AACA,0BAAoB,KAAK,cAAc;AACvC,mBAAa,YAAY;AAAA,IAC3B;AAEA,QAAI,sBAAuB,IAAE,KAAQ,MAAM,QAAQ,IAAI,IAAI,MAAO,KAAK,kBAAkB,cAAc;AAAG,mBAAa;AAEvH,aAAS,OAAO,OAAO,OAAK,EAAE,SAAS,EAAE;AAGzC,WAAO;AAEP,aAAS,eAAe;AAEtB,YAAM,qBAAqB,MAAM,QAAQ,IAAI,IAAI;AACjD,YAAM,eAAe,MAAM,SAAS;AAEpC,UAAI,MAAM,SAAS,yBAAyB;AAC1C,gBAAQ,MAAM,UAAU,GAAG,uBAAuB;AAAA,MACpD;AACA,aAAO,KAAK,EAAE,MAAM,MAAM,KAAK,GAAG,MAAM,YAAY,QAAQ,aAAa,CAAC;AAAA,IAC5E;AAAA,EACF;AAAA;AAAA,EAEA,MAAM,gBAAgB,MAAM,SAAO,CAAC,GAAG;AACrC,aAAS;AAAA,MACP,OAAO;AAAA,MACP,gBAAgB;AAAA,MAChB,WAAW;AAAA,MACX,GAAG;AAAA,IACL;AAEA,QAAI,KAAK,QAAQ,GAAG,IAAI,GAAG;AACzB,cAAQ,IAAI,uBAAqB,IAAI;AACrC,aAAO;AAAA,IACT;AACA,QAAI,QAAQ,CAAC;AACb,QAAI,iBAAiB,KAAK,MAAM,GAAG,EAAE,MAAM,CAAC;AAE5C,QAAI,qBAAqB;AACzB,QAAG,eAAe,eAAe,SAAO,CAAC,EAAE,QAAQ,GAAG,IAAI,IAAI;AAE5D,2BAAqB,SAAS,eAAe,eAAe,SAAO,CAAC,EAAE,MAAM,GAAG,EAAE,CAAC,EAAE,QAAQ,KAAK,EAAE,CAAC;AAEpG,qBAAe,eAAe,SAAO,CAAC,IAAI,eAAe,eAAe,SAAO,CAAC,EAAE,MAAM,GAAG,EAAE,CAAC;AAAA,IAChG;AACA,QAAI,iBAAiB,CAAC;AACtB,QAAI,mBAAmB;AACvB,QAAI,aAAa;AACjB,QAAI,IAAI;AAER,UAAM,YAAY,KAAK,MAAM,GAAG,EAAE,CAAC;AAEnC,UAAM,OAAO,KAAK,IAAI,MAAM,sBAAsB,SAAS;AAC3D,QAAG,EAAE,gBAAgB,SAAS,QAAQ;AACpC,cAAQ,IAAI,iBAAe,SAAS;AACpC,aAAO;AAAA,IACT;AAEA,UAAM,gBAAgB,MAAM,KAAK,IAAI,MAAM,WAAW,IAAI;AAE1D,UAAM,QAAQ,cAAc,MAAM,IAAI;AAEtC,QAAI,UAAU;AACd,SAAK,IAAI,GAAG,IAAI,MAAM,QAAQ,KAAK;AAEjC,YAAM,OAAO,MAAM,CAAC;AAEpB,UAAG,KAAK,QAAQ,KAAK,MAAM,GAAG;AAC5B,kBAAU,CAAC;AAAA,MACb;AAEA,UAAG,SAAS;AACV;AAAA,MACF;AAEA,UAAG,CAAC,MAAM,QAAQ,EAAE,QAAQ,IAAI,IAAI;AAAI;AAIxC,UAAI,CAAC,KAAK,WAAW,GAAG,KAAM,CAAC,KAAI,GAAG,EAAE,QAAQ,KAAK,CAAC,CAAC,IAAI,GAAG;AAC5D;AAAA,MACF;AAMA,YAAM,eAAe,KAAK,QAAQ,MAAM,EAAE,EAAE,KAAK;AAEjD,YAAM,gBAAgB,eAAe,QAAQ,YAAY;AACzD,UAAI,gBAAgB;AAAG;AAEvB,UAAI,eAAe,WAAW;AAAe;AAE7C,qBAAe,KAAK,YAAY;AAEhC,UAAI,eAAe,WAAW,eAAe,QAAQ;AAEnD,YAAG,uBAAuB,GAAG;AAE3B,uBAAa,IAAI;AACjB;AAAA,QACF;AAEA,YAAG,qBAAqB,oBAAmB;AACzC,uBAAa,IAAI;AACjB;AAAA,QACF;AACA;AAEA,uBAAe,IAAI;AACnB;AAAA,MACF;AAAA,IACF;AAEA,QAAI,eAAe;AAAG,aAAO;AAE7B,cAAU;AAEV,QAAI,aAAa;AACjB,SAAK,IAAI,YAAY,IAAI,MAAM,QAAQ,KAAK;AAC1C,UAAI,OAAO,eAAe,YAAc,MAAM,SAAS,YAAY;AACjE,cAAM,KAAK,KAAK;AAChB;AAAA,MACF;AACA,UAAI,OAAO,MAAM,CAAC;AAClB,UAAK,KAAK,QAAQ,GAAG,MAAM,KAAO,CAAC,KAAI,GAAG,EAAE,QAAQ,KAAK,CAAC,CAAC,MAAM,IAAI;AACnE;AAAA,MACF;AAGA,UAAI,OAAO,aAAa,aAAa,OAAO,WAAW;AACrD,cAAM,KAAK,KAAK;AAChB;AAAA,MACF;AAEA,UAAI,OAAO,aAAe,KAAK,SAAS,aAAc,OAAO,WAAY;AACvE,cAAM,gBAAgB,OAAO,YAAY;AACzC,eAAO,KAAK,MAAM,GAAG,aAAa,IAAI;AACtC;AAAA,MACF;AAGA,UAAI,KAAK,WAAW;AAAG;AAEvB,UAAI,OAAO,kBAAkB,KAAK,SAAS,OAAO,gBAAgB;AAChE,eAAO,KAAK,MAAM,GAAG,OAAO,cAAc,IAAI;AAAA,MAChD;AAEA,UAAI,KAAK,WAAW,KAAK,GAAG;AAC1B,kBAAU,CAAC;AACX;AAAA,MACF;AACA,UAAI,SAAQ;AAEV,eAAO,MAAK;AAAA,MACd;AAEA,YAAM,KAAK,IAAI;AAEf,oBAAc,KAAK;AAAA,IACrB;AAEA,QAAI,SAAS;AACX,YAAM,KAAK,KAAK;AAAA,IAClB;AACA,WAAO,MAAM,KAAK,IAAI,EAAE,KAAK;AAAA,EAC/B;AAAA;AAAA,EAGA,MAAM,eAAe,MAAM,SAAO,CAAC,GAAG;AACpC,aAAS;AAAA,MACP,OAAO;AAAA,MACP,WAAW;AAAA,MACX,gBAAgB;AAAA,MAChB,GAAG;AAAA,IACL;AACA,UAAM,YAAY,KAAK,IAAI,MAAM,sBAAsB,IAAI;AAE3D,QAAI,EAAE,qBAAqB,SAAS;AAAgB,aAAO;AAE3D,UAAM,eAAe,MAAM,KAAK,IAAI,MAAM,WAAW,SAAS;AAC9D,UAAM,aAAa,aAAa,MAAM,IAAI;AAC1C,QAAI,kBAAkB,CAAC;AACvB,QAAI,UAAU;AACd,QAAI,aAAa;AACjB,UAAMC,cAAa,OAAO,SAAS,WAAW;AAC9C,aAAS,IAAI,GAAG,gBAAgB,SAASA,aAAY,KAAK;AACxD,UAAI,OAAO,WAAW,CAAC;AAEvB,UAAI,OAAO,SAAS;AAClB;AAEF,UAAI,KAAK,WAAW;AAClB;AAEF,UAAI,OAAO,kBAAkB,KAAK,SAAS,OAAO,gBAAgB;AAChE,eAAO,KAAK,MAAM,GAAG,OAAO,cAAc,IAAI;AAAA,MAChD;AAEA,UAAI,SAAS;AACX;AAEF,UAAI,CAAC,MAAM,QAAQ,EAAE,QAAQ,IAAI,IAAI;AACnC;AAEF,UAAI,KAAK,QAAQ,KAAK,MAAM,GAAG;AAC7B,kBAAU,CAAC;AACX;AAAA,MACF;AAEA,UAAI,OAAO,aAAa,aAAa,OAAO,WAAW;AACrD,wBAAgB,KAAK,KAAK;AAC1B;AAAA,MACF;AACA,UAAI,SAAS;AAEX,eAAO,MAAO;AAAA,MAChB;AAEA,UAAI,gBAAgB,IAAI,GAAG;AAIzB,YAAK,gBAAgB,SAAS,KAAM,gBAAgB,gBAAgB,gBAAgB,SAAS,CAAC,CAAC,GAAG;AAEhG,0BAAgB,IAAI;AAAA,QACtB;AAAA,MACF;AAEA,sBAAgB,KAAK,IAAI;AAEzB,oBAAc,KAAK;AAAA,IACrB;AAEA,aAAS,IAAI,GAAG,IAAI,gBAAgB,QAAQ,KAAK;AAE/C,UAAI,gBAAgB,gBAAgB,CAAC,CAAC,GAAG;AAEvC,YAAI,MAAM,gBAAgB,SAAS,GAAG;AAEpC,0BAAgB,IAAI;AACpB;AAAA,QACF;AAEA,wBAAgB,CAAC,IAAI,gBAAgB,CAAC,EAAE,QAAQ,MAAM,EAAE;AACxD,wBAAgB,CAAC,IAAI;AAAA,EAAK,gBAAgB,CAAC;AAAA,MAC7C;AAAA,IACF;AAEA,sBAAkB,gBAAgB,KAAK,IAAI;AAC3C,WAAO;AAAA,EACT;AAAA;AAAA,EAGA,kBAAkB,gBAAgB;AAChC,QAAI,QAAQ;AACZ,QAAI,KAAK,kBAAkB,SAAS,GAAG;AACrC,eAAS,IAAI,GAAG,IAAI,KAAK,kBAAkB,QAAQ,KAAK;AACtD,YAAI,eAAe,QAAQ,KAAK,kBAAkB,CAAC,CAAC,IAAI,IAAI;AAC1D,kBAAQ;AACR,eAAK,cAAc,cAAY,KAAK,kBAAkB,CAAC,CAAC;AACxD;AAAA,QACF;AAAA,MACF;AAAA,IACF;AACA,WAAO;AAAA,EACT;AAAA;AAAA,EAEA,aAAa,WAAW,WAAS,WAAW;AAE1C,QAAI,cAAc,OAAO;AACvB,YAAM,YAAY,OAAO,KAAK,KAAK,WAAW;AAC9C,eAAS,IAAI,GAAG,IAAI,UAAU,QAAQ,KAAK;AACzC,aAAK,aAAa,KAAK,YAAY,UAAU,CAAC,CAAC,GAAG,UAAU,CAAC,CAAC;AAAA,MAChE;AACA;AAAA,IACF;AAEA,SAAK,YAAY,QAAQ,IAAI;AAE7B,QAAI,KAAK,YAAY,QAAQ,EAAE,cAAc,WAAW,GAAG;AACzD,WAAK,YAAY,QAAQ,EAAE,cAAc,WAAW,EAAE,OAAO;AAAA,IAC/D;AACA,UAAM,kBAAkB,KAAK,YAAY,QAAQ,EAAE,SAAS,OAAO,EAAE,KAAK,WAAW,CAAC;AAGtF,aAAS,QAAQ,iBAAiB,mBAAmB;AACrD,UAAM,UAAU,gBAAgB,SAAS,GAAG;AAC5C,QAAI,OAAO;AACX,QAAI,OAAO,CAAC;AAEZ,QAAI,KAAK,kBAAkB;AACzB,aAAO;AACP,aAAO;AAAA,QACL,OAAO;AAAA,MACT;AAAA,IACF;AACA,YAAQ,SAAS,KAAK;AAAA,MACpB,KAAK;AAAA,MACL;AAAA,MACA,MAAM;AAAA,MACN,QAAQ;AAAA,MACR;AAAA,IACF,CAAC;AAAA,EACH;AAAA;AAAA,EAIA,MAAM,eAAe,WAAW,SAAS;AACvC,QAAI;AAEJ,QAAI,UAAU,SAAS,SAAS,KAAO,UAAU,SAAS,CAAC,EAAE,UAAU,SAAS,SAAS,GAAG;AAC1F,aAAO,UAAU,SAAS,CAAC;AAAA,IAC7B;AAEA,QAAI,MAAM;AACR,WAAK,MAAM;AAAA,IACb,OAAO;AAEL,aAAO,UAAU,SAAS,OAAO,EAAE,KAAK,UAAU,CAAC;AAAA,IACrD;AACA,QAAI,sBAAsB;AAE1B,QAAG,CAAC,KAAK,SAAS;AAAe,6BAAuB;AAGxD,QAAG,CAAC,KAAK,SAAS,uBAAuB;AAEvC,eAAS,IAAI,GAAG,IAAI,QAAQ,QAAQ,KAAK;AAKvC,YAAI,OAAO,QAAQ,CAAC,EAAE,SAAS,UAAU;AACvC,gBAAMC,QAAO,KAAK,SAAS,OAAO,EAAE,KAAK,gBAAgB,CAAC;AAC1D,gBAAMC,QAAOD,MAAK,SAAS,KAAK;AAAA,YAC9B,KAAK;AAAA,YACL,MAAM,QAAQ,CAAC,EAAE,KAAK;AAAA,YACtB,OAAO,QAAQ,CAAC,EAAE,KAAK;AAAA,UACzB,CAAC;AACD,UAAAC,MAAK,YAAY,KAAK,yBAAyB,QAAQ,CAAC,EAAE,IAAI;AAC9D,UAAAD,MAAK,QAAQ,aAAa,MAAM;AAChC;AAAA,QACF;AAKA,YAAI;AACJ,cAAM,sBAAsB,KAAK,MAAM,QAAQ,CAAC,EAAE,aAAa,GAAG,IAAI;AACtE,YAAG,KAAK,SAAS,gBAAgB;AAC/B,gBAAM,MAAM,QAAQ,CAAC,EAAE,KAAK,MAAM,GAAG;AACrC,2BAAiB,IAAI,IAAI,SAAS,CAAC;AACnC,gBAAM,OAAO,IAAI,MAAM,GAAG,IAAI,SAAS,CAAC,EAAE,KAAK,GAAG;AAElD,2BAAiB,UAAU,yBAAyB,UAAU;AAAA,QAChE,OAAK;AACH,2BAAiB,YAAY,sBAAsB,QAAQ,QAAQ,CAAC,EAAE,KAAK,MAAM,GAAG,EAAE,IAAI,IAAI;AAAA,QAChG;AAGA,YAAG,CAAC,KAAK,qBAAqB,QAAQ,CAAC,EAAE,IAAI,GAAE;AAC7C,gBAAMA,QAAO,KAAK,SAAS,OAAO,EAAE,KAAK,gBAAgB,CAAC;AAC1D,gBAAMC,QAAOD,MAAK,SAAS,KAAK;AAAA,YAC9B,KAAK;AAAA,YACL,MAAM,QAAQ,CAAC,EAAE;AAAA,UACnB,CAAC;AACD,UAAAC,MAAK,YAAY;AAEjB,UAAAD,MAAK,QAAQ,aAAa,MAAM;AAEhC,eAAK,mBAAmBC,OAAM,QAAQ,CAAC,GAAGD,KAAI;AAC9C;AAAA,QACF;AAGA,yBAAiB,eAAe,QAAQ,OAAO,EAAE,EAAE,QAAQ,MAAM,KAAK;AAEtE,cAAM,OAAO,KAAK,SAAS,OAAO,EAAE,KAAK,oBAAoB,CAAC;AAE9D,cAAM,SAAS,KAAK,SAAS,QAAQ,EAAE,KAAK,eAAe,CAAC;AAE5D,iBAAS,QAAQ,QAAQ,gBAAgB;AACzC,cAAM,OAAO,OAAO,SAAS,KAAK;AAAA,UAChC,KAAK;AAAA,UACL,OAAO,QAAQ,CAAC,EAAE;AAAA,QACpB,CAAC;AACD,aAAK,YAAY;AAEjB,aAAK,mBAAmB,MAAM,QAAQ,CAAC,GAAG,IAAI;AAC9C,eAAO,iBAAiB,SAAS,CAAC,UAAU;AAE1C,cAAI,SAAS,MAAM,OAAO;AAC1B,iBAAO,CAAC,OAAO,UAAU,SAAS,eAAe,GAAG;AAClD,qBAAS,OAAO;AAAA,UAClB;AAEA,iBAAO,UAAU,OAAO,cAAc;AAAA,QACxC,CAAC;AACD,cAAM,WAAW,KAAK,SAAS,MAAM,EAAE,KAAK,GAAG,CAAC;AAChD,cAAM,qBAAqB,SAAS,SAAS,MAAM;AAAA,UACjD,KAAK;AAAA,UACL,OAAO,QAAQ,CAAC,EAAE;AAAA,QACpB,CAAC;AACD,YAAG,QAAQ,CAAC,EAAE,KAAK,QAAQ,GAAG,IAAI,IAAG;AACnC,mBAAS,iBAAiB,eAAgB,MAAM,KAAK,gBAAgB,QAAQ,CAAC,EAAE,MAAM,EAAC,OAAO,IAAI,WAAW,IAAI,CAAC,GAAI,oBAAoB,QAAQ,CAAC,EAAE,MAAM,IAAI,SAAS,UAAU,CAAC;AAAA,QACrL,OAAK;AACH,gBAAM,kBAAkB,MAAM,KAAK,eAAe,QAAQ,CAAC,EAAE,MAAM,EAAC,OAAO,IAAI,WAAW,IAAI,CAAC;AAC/F,cAAG,CAAC;AAAiB;AACrB,mBAAS,iBAAiB,eAAe,iBAAiB,oBAAoB,QAAQ,CAAC,EAAE,MAAM,IAAI,SAAS,UAAU,CAAC;AAAA,QACzH;AACA,aAAK,mBAAmB,UAAU,QAAQ,CAAC,GAAG,IAAI;AAAA,MACpD;AACA,WAAK,aAAa,WAAW,OAAO;AACpC;AAAA,IACF;AAGA,UAAM,kBAAkB,CAAC;AACzB,aAAS,IAAI,GAAG,IAAI,QAAQ,QAAQ,KAAK;AACvC,YAAM,OAAO,QAAQ,CAAC;AACtB,YAAM,OAAO,KAAK;AAElB,UAAI,OAAO,SAAS,UAAU;AAC5B,wBAAgB,KAAK,IAAI,IAAI,CAAC,IAAI;AAClC;AAAA,MACF;AACA,UAAI,KAAK,QAAQ,GAAG,IAAI,IAAI;AAC1B,cAAM,YAAY,KAAK,MAAM,GAAG,EAAE,CAAC;AACnC,YAAI,CAAC,gBAAgB,SAAS,GAAG;AAC/B,0BAAgB,SAAS,IAAI,CAAC;AAAA,QAChC;AACA,wBAAgB,SAAS,EAAE,KAAK,QAAQ,CAAC,CAAC;AAAA,MAC5C,OAAO;AACL,YAAI,CAAC,gBAAgB,IAAI,GAAG;AAC1B,0BAAgB,IAAI,IAAI,CAAC;AAAA,QAC3B;AAEA,wBAAgB,IAAI,EAAE,QAAQ,QAAQ,CAAC,CAAC;AAAA,MAC1C;AAAA,IACF;AAEA,UAAM,OAAO,OAAO,KAAK,eAAe;AACxC,aAAS,IAAI,GAAG,IAAI,KAAK,QAAQ,KAAK;AACpC,YAAM,OAAO,gBAAgB,KAAK,CAAC,CAAC;AAKpC,UAAI,OAAO,KAAK,CAAC,EAAE,SAAS,UAAU;AACpC,cAAM,OAAO,KAAK,CAAC;AACnB,cAAM,OAAO,KAAK;AAClB,YAAI,KAAK,KAAK,WAAW,MAAM,GAAG;AAChC,gBAAMA,QAAO,KAAK,SAAS,OAAO,EAAE,KAAK,gBAAgB,CAAC;AAC1D,gBAAM,OAAOA,MAAK,SAAS,KAAK;AAAA,YAC9B,KAAK;AAAA,YACL,MAAM,KAAK;AAAA,YACX,OAAO,KAAK;AAAA,UACd,CAAC;AACD,eAAK,YAAY,KAAK,yBAAyB,IAAI;AACnD,UAAAA,MAAK,QAAQ,aAAa,MAAM;AAChC;AAAA,QACF;AAAA,MACF;AAIA,UAAI;AACJ,YAAM,sBAAsB,KAAK,MAAM,KAAK,CAAC,EAAE,aAAa,GAAG,IAAI;AACnE,UAAI,KAAK,SAAS,gBAAgB;AAChC,cAAM,MAAM,KAAK,CAAC,EAAE,KAAK,MAAM,GAAG;AAClC,yBAAiB,IAAI,IAAI,SAAS,CAAC;AACnC,cAAM,OAAO,IAAI,MAAM,GAAG,IAAI,SAAS,CAAC,EAAE,KAAK,GAAG;AAClD,yBAAiB,UAAU,UAAU,kCAAkC;AAAA,MACzE,OAAO;AACL,yBAAiB,KAAK,CAAC,EAAE,KAAK,MAAM,GAAG,EAAE,IAAI;AAE7C,0BAAkB,QAAQ;AAAA,MAC5B;AAMA,UAAG,CAAC,KAAK,qBAAqB,KAAK,CAAC,EAAE,IAAI,GAAG;AAC3C,cAAMA,QAAO,KAAK,SAAS,OAAO,EAAE,KAAK,gBAAgB,CAAC;AAC1D,cAAME,aAAYF,MAAK,SAAS,KAAK;AAAA,UACnC,KAAK;AAAA,UACL,OAAO,KAAK,CAAC,EAAE;AAAA,QACjB,CAAC;AACD,QAAAE,WAAU,YAAY;AAEtB,aAAK,mBAAmBA,YAAW,KAAK,CAAC,GAAGF,KAAI;AAChD;AAAA,MACF;AAIA,uBAAiB,eAAe,QAAQ,OAAO,EAAE,EAAE,QAAQ,MAAM,KAAK;AACtE,YAAM,OAAO,KAAK,SAAS,OAAO,EAAE,KAAK,oBAAoB,CAAC;AAC9D,YAAM,SAAS,KAAK,SAAS,QAAQ,EAAE,KAAK,eAAe,CAAC;AAE5D,eAAS,QAAQ,QAAQ,gBAAgB;AACzC,YAAM,YAAY,OAAO,SAAS,KAAK;AAAA,QACrC,KAAK;AAAA,QACL,OAAO,KAAK,CAAC,EAAE;AAAA,MACjB,CAAC;AACD,gBAAU,YAAY;AAEtB,WAAK,mBAAmB,WAAW,KAAK,CAAC,GAAG,MAAM;AAClD,aAAO,iBAAiB,SAAS,CAAC,UAAU;AAE1C,YAAI,SAAS,MAAM;AACnB,eAAO,CAAC,OAAO,UAAU,SAAS,eAAe,GAAG;AAClD,mBAAS,OAAO;AAAA,QAClB;AACA,eAAO,UAAU,OAAO,cAAc;AAAA,MAExC,CAAC;AACD,YAAM,iBAAiB,KAAK,SAAS,IAAI;AAEzC,eAAS,IAAI,GAAG,IAAI,KAAK,QAAQ,KAAK;AAEpC,YAAG,KAAK,CAAC,EAAE,KAAK,QAAQ,GAAG,IAAI,IAAI;AACjC,gBAAM,QAAQ,KAAK,CAAC;AACpB,gBAAM,aAAa,eAAe,SAAS,MAAM;AAAA,YAC/C,KAAK;AAAA,YACL,OAAO,MAAM;AAAA,UACf,CAAC;AAED,cAAG,KAAK,SAAS,GAAG;AAClB,kBAAM,gBAAgB,KAAK,qBAAqB,KAAK;AACrD,kBAAM,uBAAuB,KAAK,MAAM,MAAM,aAAa,GAAG,IAAI;AAClE,uBAAW,YAAY,UAAU,mBAAmB;AAAA,UACtD;AACA,gBAAM,kBAAkB,WAAW,SAAS,KAAK;AAEjD,mBAAS,iBAAiB,eAAgB,MAAM,KAAK,gBAAgB,MAAM,MAAM,EAAC,OAAO,IAAI,WAAW,IAAI,CAAC,GAAI,iBAAiB,MAAM,MAAM,IAAI,SAAS,UAAU,CAAC;AAEtK,eAAK,mBAAmB,YAAY,OAAO,cAAc;AAAA,QAC3D,OAAK;AAEH,gBAAMG,kBAAiB,KAAK,SAAS,IAAI;AACzC,gBAAM,aAAaA,gBAAe,SAAS,MAAM;AAAA,YAC/C,KAAK;AAAA,YACL,OAAO,KAAK,CAAC,EAAE;AAAA,UACjB,CAAC;AACD,gBAAM,kBAAkB,WAAW,SAAS,KAAK;AACjD,cAAI,kBAAkB,MAAM,KAAK,eAAe,KAAK,CAAC,EAAE,MAAM,EAAC,OAAO,IAAI,WAAW,IAAI,CAAC;AAC1F,cAAG,CAAC;AAAiB;AACrB,mBAAS,iBAAiB,eAAe,iBAAiB,iBAAiB,KAAK,CAAC,EAAE,MAAM,IAAI,SAAS,UAAU,CAAC;AACjH,eAAK,mBAAmB,YAAY,KAAK,CAAC,GAAGA,eAAc;AAAA,QAE7D;AAAA,MACF;AAAA,IACF;AACA,SAAK,aAAa,WAAW,MAAM;AAAA,EACrC;AAAA,EAEA,mBAAmB,MAAM,MAAM,MAAM;AACnC,SAAK,iBAAiB,SAAS,OAAO,UAAU;AAC9C,YAAM,KAAK,UAAU,MAAM,KAAK;AAAA,IAClC,CAAC;AAGD,SAAK,QAAQ,aAAa,MAAM;AAChC,SAAK,iBAAiB,aAAa,CAAC,UAAU;AAC5C,YAAM,cAAc,KAAK,IAAI;AAC7B,YAAM,YAAY,KAAK,KAAK,MAAM,GAAG,EAAE,CAAC;AACxC,YAAM,OAAO,KAAK,IAAI,cAAc,qBAAqB,WAAW,EAAE;AACtE,YAAM,WAAW,YAAY,SAAS,OAAO,IAAI;AAEjD,kBAAY,YAAY,OAAO,QAAQ;AAAA,IACzC,CAAC;AAED,QAAI,KAAK,KAAK,QAAQ,GAAG,IAAI;AAAI;AAEjC,SAAK,iBAAiB,aAAa,CAAC,UAAU;AAC5C,WAAK,IAAI,UAAU,QAAQ,cAAc;AAAA,QACvC;AAAA,QACA,QAAQ;AAAA,QACR,aAAa;AAAA,QACb,UAAU;AAAA,QACV,UAAU,KAAK;AAAA,MACjB,CAAC;AAAA,IACH,CAAC;AAAA,EACH;AAAA;AAAA;AAAA,EAIA,MAAM,UAAU,MAAM,QAAM,MAAM;AAChC,QAAI;AACJ,QAAI;AACJ,QAAI,KAAK,KAAK,QAAQ,GAAG,IAAI,IAAI;AAE/B,mBAAa,KAAK,IAAI,cAAc,qBAAqB,KAAK,KAAK,MAAM,GAAG,EAAE,CAAC,GAAG,EAAE;AAEpF,YAAM,oBAAoB,KAAK,IAAI,cAAc,aAAa,UAAU;AAGxE,UAAI,eAAe,KAAK,KAAK,MAAM,GAAG,EAAE,IAAI;AAE5C,UAAI,YAAY;AAChB,UAAI,aAAa,QAAQ,GAAG,IAAI,IAAI;AAElC,oBAAY,SAAS,aAAa,MAAM,GAAG,EAAE,CAAC,EAAE,MAAM,GAAG,EAAE,CAAC,CAAC;AAE7D,uBAAe,aAAa,MAAM,GAAG,EAAE,CAAC;AAAA,MAC1C;AAEA,YAAM,WAAW,kBAAkB;AAEnC,eAAQ,IAAI,GAAG,IAAI,SAAS,QAAQ,KAAK;AACvC,YAAI,SAAS,CAAC,EAAE,YAAY,cAAc;AAExC,cAAG,cAAc,GAAG;AAClB,sBAAU,SAAS,CAAC;AACpB;AAAA,UACF;AACA;AAAA,QACF;AAAA,MACF;AAAA,IAEF,OAAO;AACL,mBAAa,KAAK,IAAI,cAAc,qBAAqB,KAAK,MAAM,EAAE;AAAA,IACxE;AACA,QAAI;AACJ,QAAG,OAAO;AAER,YAAM,MAAM,SAAS,OAAO,WAAW,KAAK;AAE5C,aAAO,KAAK,IAAI,UAAU,QAAQ,GAAG;AAAA,IACvC,OAAK;AAEH,aAAO,KAAK,IAAI,UAAU,kBAAkB;AAAA,IAC9C;AACA,UAAM,KAAK,SAAS,UAAU;AAC9B,QAAI,SAAS;AACX,UAAI,EAAE,OAAO,IAAI,KAAK;AACtB,YAAM,MAAM,EAAE,MAAM,QAAQ,SAAS,MAAM,MAAM,IAAI,EAAE;AACvD,aAAO,UAAU,GAAG;AACpB,aAAO,eAAe,EAAE,IAAI,KAAK,MAAM,IAAI,GAAG,IAAI;AAAA,IACpD;AAAA,EACF;AAAA,EAEA,qBAAqB,OAAO;AAC1B,UAAM,iBAAiB,MAAM,KAAK,MAAM,KAAK,EAAE,CAAC,EAAE,MAAM,GAAG;AAE3D,QAAI,gBAAgB;AACpB,aAAS,IAAI,eAAe,SAAS,GAAG,KAAK,GAAG,KAAK;AACnD,UAAG,cAAc,SAAS,GAAG;AAC3B,wBAAgB,MAAM;AAAA,MACxB;AACA,sBAAgB,eAAe,CAAC,IAAI;AAEpC,UAAI,cAAc,SAAS,KAAK;AAC9B;AAAA,MACF;AAAA,IACF;AAEA,QAAI,cAAc,WAAW,KAAK,GAAG;AACnC,sBAAgB,cAAc,MAAM,CAAC;AAAA,IACvC;AACA,WAAO;AAAA,EAET;AAAA,EAEA,qBAAqB,MAAM;AACzB,WAAQ,KAAK,QAAQ,KAAK,MAAM,MAAQ,KAAK,QAAQ,aAAa,MAAM;AAAA,EAC1E;AAAA,EAEA,yBAAyB,MAAK;AAC5B,QAAG,KAAK,QAAQ;AACd,UAAG,KAAK,WAAW;AAAS,aAAK,SAAS;AAC1C,aAAO,UAAU,KAAK,qBAAqB,KAAK;AAAA,IAClD;AAEA,QAAI,SAAS,KAAK,KAAK,QAAQ,iBAAiB,EAAE;AAElD,aAAS,OAAO,MAAM,GAAG,EAAE,CAAC;AAE5B,WAAO,oBAAa,qBAAqB,KAAK;AAAA,EAChD;AAAA;AAAA,EAEA,MAAM,kBAAkB;AACtB,QAAG,CAAC,KAAK,WAAW,KAAK,QAAQ,WAAW,GAAE;AAC5C,WAAK,UAAU,MAAM,KAAK,YAAY;AAAA,IACxC;AACA,WAAO,KAAK;AAAA,EACd;AAAA;AAAA,EAEA,MAAM,YAAY,OAAO,KAAK;AAC5B,QAAI,WAAW,MAAM,KAAK,IAAI,MAAM,QAAQ,KAAK,IAAI,GAAG;AACxD,QAAI,cAAc,CAAC;AACnB,aAAS,IAAI,GAAG,IAAI,QAAQ,QAAQ,KAAK;AACvC,UAAI,QAAQ,CAAC,EAAE,WAAW,GAAG;AAAG;AAChC,kBAAY,KAAK,QAAQ,CAAC,CAAC;AAC3B,oBAAc,YAAY,OAAO,MAAM,KAAK,YAAY,QAAQ,CAAC,IAAI,GAAG,CAAC;AAAA,IAC3E;AACA,WAAO;AAAA,EACT;AAAA,EAGA,MAAM,aAAa;AAEjB,QAAG,CAAC,KAAK,SAAS,aAAY;AAC5B,UAAI,SAAS,OAAO,kGAAkG;AACtH;AAAA,IACF;AACA,YAAQ,IAAI,eAAe;AAE3B,UAAM,QAAQ,KAAK,IAAI,MAAM,iBAAiB,EAAE,OAAO,CAAC,SAAS;AAE/D,eAAQ,IAAI,GAAG,IAAI,KAAK,gBAAgB,QAAQ,KAAK;AACnD,YAAG,KAAK,KAAK,QAAQ,KAAK,gBAAgB,CAAC,CAAC,IAAI,IAAI;AAClD,iBAAO;AAAA,QACT;AAAA,MACF;AACA,aAAO;AAAA,IACT,CAAC;AACD,UAAM,QAAQ,MAAM,KAAK,mBAAmB,KAAK;AACjD,YAAQ,IAAI,cAAc;AAE1B,UAAM,KAAK,IAAI,MAAM,QAAQ,MAAM,iCAAiC,KAAK,UAAU,OAAO,MAAM,CAAC,CAAC;AAClG,YAAQ,IAAI,aAAa;AACzB,YAAQ,IAAI,KAAK,SAAS,WAAW;AAErC,UAAM,WAAW,OAAO,GAAG,SAAS,YAAY;AAAA,MAC9C,KAAK;AAAA,MACL,QAAQ;AAAA,MACR,SAAS;AAAA,QACP,gBAAgB;AAAA,MAClB;AAAA,MACA,aAAa;AAAA,MACb,MAAM,KAAK,UAAU;AAAA,QACnB,aAAa,KAAK,SAAS;AAAA,QAC3B;AAAA,MACF,CAAC;AAAA,IACH,CAAC;AACD,YAAQ,IAAI,QAAQ;AAAA,EAEtB;AAAA,EAEA,MAAM,mBAAmB,OAAO;AAC9B,QAAI,SAAS,CAAC;AAEd,aAAQ,IAAI,GAAG,IAAI,MAAM,QAAQ,KAAK;AACpC,UAAI,OAAO,MAAM,CAAC;AAClB,UAAI,QAAQ,KAAK,KAAK,MAAM,GAAG;AAC/B,UAAI,UAAU;AAEd,eAAS,KAAK,GAAG,KAAK,MAAM,QAAQ,MAAM;AACxC,YAAI,OAAO,MAAM,EAAE;AAEnB,YAAI,OAAO,MAAM,SAAS,GAAG;AAE3B,kBAAQ,IAAI,IAAI,MAAM,KAAK,IAAI,MAAM,WAAW,IAAI;AAAA,QACtD,OAAO;AAEL,cAAI,CAAC,QAAQ,IAAI,GAAG;AAClB,oBAAQ,IAAI,IAAI,CAAC;AAAA,UACnB;AAEA,oBAAU,QAAQ,IAAI;AAAA,QACxB;AAAA,MACF;AAAA,IACF;AAEA,WAAO;AAAA,EACT;AAEF;AAEA,IAAM,8BAA8B;AACpC,IAAM,uBAAN,cAAmC,SAAS,SAAS;AAAA,EACnD,YAAY,MAAM,QAAQ;AACxB,UAAM,IAAI;AACV,SAAK,SAAS;AACd,SAAK,UAAU;AACf,SAAK,YAAY;AAAA,EACnB;AAAA,EACA,cAAc;AACZ,WAAO;AAAA,EACT;AAAA,EAEA,iBAAiB;AACf,WAAO;AAAA,EACT;AAAA,EAEA,UAAU;AACR,WAAO;AAAA,EACT;AAAA,EAGA,YAAY,SAAS;AACnB,UAAM,YAAY,KAAK,YAAY,SAAS,CAAC;AAE7C,cAAU,MAAM;AAEhB,SAAK,iBAAiB,SAAS;AAE/B,QAAI,MAAM,QAAQ,OAAO,GAAG;AAC1B,eAAS,IAAI,GAAG,IAAI,QAAQ,QAAQ,KAAK;AACvC,kBAAU,SAAS,KAAK,EAAE,KAAK,cAAc,MAAM,QAAQ,CAAC,EAAE,CAAC;AAAA,MACjE;AAAA,IACF,OAAK;AAEH,gBAAU,SAAS,KAAK,EAAE,KAAK,cAAc,MAAM,QAAQ,CAAC;AAAA,IAC9D;AAAA,EACF;AAAA,EACA,iBAAiB,MAAM,iBAAe,OAAO;AAK3C,QAAI,CAAC,gBAAgB;AACnB,aAAO,KAAK,MAAM,GAAG,EAAE,IAAI;AAAA,IAC7B;AAEA,QAAI,KAAK,QAAQ,GAAG,IAAI,IAAI;AAE1B,aAAO,KAAK,MAAM,KAAK;AAEvB,WAAK,CAAC,IAAI,UAAU,KAAK,CAAC;AAE1B,aAAO,KAAK,KAAK,EAAE;AAEnB,aAAO,KAAK,QAAQ,OAAO,QAAK;AAAA,IAClC,OAAK;AAEH,aAAO,KAAK,QAAQ,OAAO,EAAE;AAAA,IAC/B;AACA,WAAO;AAAA,EACT;AAAA,EAGA,YAAY,SAAS,kBAAgB,MAAM,eAAa,OAAO;AAE7D,UAAM,YAAY,KAAK,YAAY,SAAS,CAAC;AAE7C,QAAG,CAAC,cAAa;AAEf,gBAAU,MAAM;AAChB,WAAK,iBAAiB,WAAW,eAAe;AAAA,IAClD;AAEA,SAAK,OAAO,eAAe,WAAW,OAAO;AAAA,EAC/C;AAAA,EAEA,iBAAiB,WAAW,kBAAgB,MAAM;AAChD,QAAI;AAEJ,QAAK,UAAU,SAAS,SAAS,KAAO,UAAU,SAAS,CAAC,EAAE,UAAU,SAAS,YAAY,GAAI;AAC/F,gBAAU,UAAU,SAAS,CAAC;AAC9B,cAAQ,MAAM;AAAA,IAChB,OAAO;AAEL,gBAAU,UAAU,SAAS,OAAO,EAAE,KAAK,aAAa,CAAC;AAAA,IAC3D;AAEA,QAAI,iBAAiB;AACnB,cAAQ,SAAS,KAAK,EAAE,KAAK,cAAc,MAAM,gBAAgB,CAAC;AAAA,IACpE;AAEA,UAAM,cAAc,QAAQ,SAAS,UAAU,EAAE,KAAK,iBAAiB,CAAC;AAExE,aAAS,QAAQ,aAAa,gBAAgB;AAE9C,gBAAY,iBAAiB,SAAS,MAAM;AAE1C,WAAK,OAAO,UAAU;AAAA,IACxB,CAAC;AAED,UAAM,gBAAgB,QAAQ,SAAS,UAAU,EAAE,KAAK,mBAAmB,CAAC;AAE5E,aAAS,QAAQ,eAAe,QAAQ;AAExC,kBAAc,iBAAiB,SAAS,MAAM;AAE5C,cAAQ,MAAM;AAEd,YAAM,mBAAmB,QAAQ,SAAS,OAAO,EAAE,KAAK,yBAAyB,CAAC;AAClF,YAAM,QAAQ,iBAAiB,SAAS,SAAS;AAAA,QAC/C,KAAK;AAAA,QACL,MAAM;AAAA,QACN,aAAa;AAAA,MACf,CAAC;AAED,YAAM,MAAM;AAEZ,YAAM,iBAAiB,WAAW,CAAC,UAAU;AAE3C,YAAI,MAAM,QAAQ,UAAU;AAC1B,eAAK,oBAAoB;AAEzB,eAAK,iBAAiB,WAAW,eAAe;AAAA,QAClD;AAAA,MACF,CAAC;AAGD,YAAM,iBAAiB,SAAS,CAAC,UAAU;AAEzC,aAAK,oBAAoB;AAEzB,cAAM,cAAc,MAAM;AAE1B,YAAI,MAAM,QAAQ,WAAW,gBAAgB,IAAI;AAC/C,eAAK,OAAO,WAAW;AAAA,QACzB,WAES,gBAAgB,IAAI;AAE3B,uBAAa,KAAK,cAAc;AAEhC,eAAK,iBAAiB,WAAW,MAAM;AACrC,iBAAK,OAAO,aAAa,IAAI;AAAA,UAC/B,GAAG,GAAG;AAAA,QACR;AAAA,MACF,CAAC;AAAA,IACH,CAAC;AAAA,EACH;AAAA;AAAA,EAGA,4BAA4B;AAE1B,UAAM,YAAY,KAAK,YAAY,SAAS,CAAC;AAE7C,cAAU,MAAM;AAEhB,cAAU,SAAS,MAAM,EAAE,KAAK,aAAa,MAAM,4BAA4B,CAAC;AAEhF,UAAM,aAAa,UAAU,SAAS,OAAO,EAAE,KAAK,cAAc,CAAC;AAEnE,UAAM,gBAAgB,WAAW,SAAS,UAAU,EAAE,KAAK,YAAY,MAAM,yBAAyB,CAAC;AAEvG,eAAW,SAAS,KAAK,EAAE,KAAK,gBAAgB,MAAM,0FAA0F,CAAC;AAEjJ,UAAM,eAAe,WAAW,SAAS,UAAU,EAAE,KAAK,YAAY,MAAM,QAAQ,CAAC;AAErF,eAAW,SAAS,KAAK,EAAE,KAAK,gBAAgB,MAAM,mEAAmE,CAAC;AAG1H,kBAAc,iBAAiB,SAAS,OAAO,UAAU;AAEvD,YAAM,KAAK,OAAO,eAAe,qBAAqB;AAEtD,YAAM,KAAK,mBAAmB;AAAA,IAChC,CAAC;AAGD,iBAAa,iBAAiB,SAAS,OAAO,UAAU;AACtD,cAAQ,IAAI,uCAAuC;AAEnD,YAAM,KAAK,OAAO,UAAU;AAE5B,YAAM,KAAK,mBAAmB;AAAA,IAChC,CAAC;AAAA,EACH;AAAA,EAEA,MAAM,SAAS;AACb,UAAM,YAAY,KAAK,YAAY,SAAS,CAAC;AAC7C,cAAU,MAAM;AAEhB,cAAU,SAAS,KAAK,EAAE,KAAK,iBAAiB,MAAM,mCAAmC,CAAC;AAG1F,SAAK,OAAO,cAAc,KAAK,IAAI,UAAU,GAAG,aAAa,CAAC,SAAS;AAErE,UAAG,CAAC,MAAM;AAER;AAAA,MACF;AAEA,UAAG,qBAAqB,QAAQ,KAAK,SAAS,MAAM,IAAI;AACtD,eAAO,KAAK,YAAY;AAAA,UACtB,WAAS,KAAK;AAAA,UACb,uCAAqC,qBAAqB,KAAK,IAAI,IAAE;AAAA,QACxE,CAAC;AAAA,MACH;AAEA,UAAG,KAAK,WAAU;AAChB,qBAAa,KAAK,SAAS;AAAA,MAC7B;AACA,WAAK,YAAY,WAAW,MAAM;AAChC,aAAK,mBAAmB,IAAI;AAC5B,aAAK,YAAY;AAAA,MACnB,GAAG,GAAI;AAAA,IAET,CAAC,CAAC;AAEF,SAAK,IAAI,UAAU,wBAAwB,6BAA6B;AAAA,MACpE,SAAS;AAAA,MACT,YAAY;AAAA,IAChB,CAAC;AACD,SAAK,IAAI,UAAU,wBAAwB,kCAAkC;AAAA,MACzE,SAAS;AAAA,MACT,YAAY;AAAA,IAChB,CAAC;AAED,SAAK,IAAI,UAAU,cAAc,KAAK,WAAW,KAAK,IAAI,CAAC;AAAA,EAE7D;AAAA,EAEA,MAAM,aAAa;AACjB,SAAK,YAAY,4BAA4B;AAC7C,UAAM,gBAAgB,MAAM,KAAK,OAAO,UAAU;AAClD,QAAG,eAAc;AACf,WAAK,YAAY,yBAAyB;AAC1C,YAAM,KAAK,mBAAmB;AAAA,IAChC,OAAK;AACH,WAAK,0BAA0B;AAAA,IACjC;AAOA,SAAK,MAAM,IAAI,wBAAwB,KAAK,KAAK,KAAK,QAAQ,IAAI;AAElE,KAAC,OAAO,yBAAyB,IAAI,KAAK,QAAQ,KAAK,SAAS,MAAM,OAAO,OAAO,yBAAyB,CAAC;AAAA,EAEhH;AAAA,EAEA,MAAM,UAAU;AACd,YAAQ,IAAI,gCAAgC;AAC5C,SAAK,IAAI,UAAU,0BAA0B,2BAA2B;AACxE,SAAK,OAAO,OAAO;AAAA,EACrB;AAAA,EAEA,MAAM,mBAAmB,UAAQ,MAAM;AACrC,YAAQ,IAAI,uBAAuB;AAEnC,QAAG,CAAC,KAAK,OAAO,SAAS,SAAS;AAChC,WAAK,YAAY,yDAAyD;AAC1E;AAAA,IACF;AACA,QAAG,CAAC,KAAK,OAAO,mBAAkB;AAChC,YAAM,KAAK,OAAO,UAAU;AAAA,IAC9B;AAEA,QAAG,CAAC,KAAK,OAAO,mBAAmB;AACjC,cAAQ,IAAI,wDAAwD;AACpE,WAAK,0BAA0B;AAC/B;AAAA,IACF;AACA,SAAK,YAAY,6BAA6B;AAI9C,QAAG,OAAO,YAAY,UAAU;AAC9B,YAAM,mBAAmB;AAEzB,YAAM,KAAK,OAAO,gBAAgB;AAClC;AAAA,IACF;AAKA,SAAK,UAAU;AACf,SAAK,iBAAiB;AACtB,SAAK,YAAY;AACjB,SAAK,OAAO;AAEZ,QAAG,KAAK,UAAU;AAChB,oBAAc,KAAK,QAAQ;AAC3B,WAAK,WAAW;AAAA,IAClB;AAEA,SAAK,WAAW,YAAY,MAAM;AAChC,UAAG,CAAC,KAAK,WAAU;AACjB,YAAG,KAAK,gBAAgB,SAAS,OAAO;AACtC,eAAK,YAAY;AACjB,eAAK,wBAAwB,KAAK,IAAI;AAAA,QACxC,OAAK;AAEH,eAAK,OAAO,KAAK,IAAI,UAAU,cAAc;AAE7C,cAAG,CAAC,KAAK,QAAQ,KAAK,QAAQ,GAAG;AAC/B,0BAAc,KAAK,QAAQ;AAC3B,iBAAK,YAAY,gBAAgB;AACjC;AAAA,UACF;AAAA,QACF;AAAA,MACF,OAAK;AACH,YAAG,KAAK,SAAS;AACf,wBAAc,KAAK,QAAQ;AAE3B,cAAI,OAAO,KAAK,YAAY,UAAU;AACpC,iBAAK,YAAY,KAAK,OAAO;AAAA,UAC/B,OAAO;AAEL,iBAAK,YAAY,KAAK,SAAS,WAAW,KAAK,KAAK,IAAI;AAAA,UAC1D;AAEA,cAAI,KAAK,OAAO,WAAW,kBAAkB,SAAS,GAAG;AACvD,iBAAK,OAAO,uBAAuB;AAAA,UACrC;AAEA,eAAK,OAAO,kBAAkB;AAC9B;AAAA,QACF,OAAK;AACH,eAAK;AACL,eAAK,YAAY,gCAA8B,KAAK,cAAc;AAAA,QACpE;AAAA,MACF;AAAA,IACF,GAAG,EAAE;AAAA,EACP;AAAA,EAEA,MAAM,wBAAwB,MAAM;AAClC,SAAK,UAAU,MAAM,KAAK,OAAO,sBAAsB,IAAI;AAAA,EAC7D;AAAA,EAEA,sBAAsB;AACpB,QAAI,KAAK,gBAAgB;AACvB,mBAAa,KAAK,cAAc;AAChC,WAAK,iBAAiB;AAAA,IACxB;AAAA,EACF;AAAA,EAEA,MAAM,OAAO,aAAa,eAAa,OAAO;AAC5C,UAAM,UAAU,MAAM,KAAK,OAAO,IAAI,OAAO,WAAW;AAExD,UAAM,kBAAkB,eAAe,YAAY,SAAS,MAAM,YAAY,UAAU,GAAG,GAAG,IAAI,QAAQ;AAC1G,SAAK,YAAY,SAAS,iBAAiB,YAAY;AAAA,EACzD;AAEF;AACA,IAAM,0BAAN,MAA8B;AAAA,EAC5B,YAAY,KAAK,QAAQ,MAAM;AAC7B,SAAK,MAAM;AACX,SAAK,SAAS;AACd,SAAK,OAAO;AAAA,EACd;AAAA,EACA,MAAM,OAAQ,aAAa;AACzB,WAAO,MAAM,KAAK,OAAO,IAAI,OAAO,WAAW;AAAA,EACjD;AAAA;AAAA,EAEA,MAAM,yBAAyB;AAC7B,UAAM,KAAK,OAAO,UAAU;AAC5B,UAAM,KAAK,KAAK,mBAAmB;AAAA,EACrC;AACF;AACA,IAAM,cAAN,MAAkB;AAAA,EAChB,YAAY,KAAK,QAAQ;AACvB,SAAK,MAAM;AACX,SAAK,SAAS;AAAA,EAChB;AAAA,EACA,MAAM,OAAQ,aAAa,SAAO,CAAC,GAAG;AACpC,aAAS;AAAA,MACP,eAAe,KAAK,OAAO,SAAS;AAAA,MACpC,GAAG;AAAA,IACL;AACA,QAAI,UAAU,CAAC;AACf,UAAM,OAAO,MAAM,KAAK,OAAO,6BAA6B,WAAW;AACvE,QAAI,QAAQ,KAAK,QAAQ,KAAK,KAAK,CAAC,KAAK,KAAK,KAAK,CAAC,EAAE,WAAW;AAC/D,gBAAU,KAAK,OAAO,eAAe,QAAQ,KAAK,KAAK,CAAC,EAAE,WAAW,MAAM;AAAA,IAC7E,OAAO;AAEL,UAAI,SAAS,OAAO,4CAA4C;AAAA,IAClE;AACA,WAAO;AAAA,EACT;AACF;AAEA,IAAM,8BAAN,cAA0C,SAAS,iBAAiB;AAAA,EAClE,YAAY,KAAK,QAAQ;AACvB,UAAM,KAAK,MAAM;AACjB,SAAK,SAAS;AAAA,EAChB;AAAA,EACA,UAAU;AACR,UAAM;AAAA,MACJ;AAAA,IACF,IAAI;AACJ,gBAAY,MAAM;AAClB,gBAAY,SAAS,MAAM;AAAA,MACzB,MAAM;AAAA,IACR,CAAC;AAED,gBAAY,SAAS,KAAK;AAAA,MACxB,MAAM;AAAA,IACR,CAAC;AAED,UAAM,0BAA0B,YAAY,SAAS,IAAI;AACzD,4BAAwB,SAAS,MAAM;AAAA,MACrC,MAAM;AAAA,IACR,CAAC;AACD,4BAAwB,SAAS,MAAM;AAAA,MACrC,MAAM;AAAA,IACR,CAAC;AACD,4BAAwB,SAAS,MAAM;AAAA,MACrC,MAAM;AAAA,IACR,CAAC;AAED,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,uBAAuB,EAAE,QAAQ,sDAAsD,EAAE,QAAQ,CAAC,SAAS,KAAK,eAAe,wBAAwB,EAAE,SAAS,KAAK,OAAO,SAAS,WAAW,EAAE,SAAS,OAAO,UAAU;AACtQ,WAAK,OAAO,SAAS,cAAc,MAAM,KAAK;AAC9C,YAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IACrC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,QAAQ,EAAE,QAAQ,mIAAmI,EAAE,UAAU,CAAC,WAAW,OAAO,cAAc,mBAAmB,EAAE,QAAQ,YAAY;AACnR,YAAM,KAAK,OAAO,aAAa;AAAA,IACjC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,YAAY,EAAE,QAAQ,8GAA8G,EAAE,UAAU,CAAC,WAAW,OAAO,cAAc,YAAY,EAAE,QAAQ,YAAY;AAE3P,YAAM,KAAK,OAAO,WAAW;AAAA,IAC/B,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,oBAAoB,EAAE,QAAQ,oBAAoB,EAAE,UAAU,CAAC,WAAW,OAAO,cAAc,oBAAoB,EAAE,QAAQ,YAAY;AACjL,YAAM,gBAAgB;AAAA,QACpB;AAAA,QACA;AAAA,MACF;AACA,UAAG,CAAC,KAAK,OAAO,oBAAmB;AACjC,aAAK,OAAO,qBAAqB,KAAK,MAAM,KAAK,OAAO,CAAC;AAAA,MAC3D;AAEA,aAAO,KAAK,cAAc,KAAK,OAAO,kBAAkB,CAAC;AAAA,IAC3D,CAAC,CAAC;AAGF,gBAAY,SAAS,MAAM;AAAA,MACzB,MAAM;AAAA,IACR,CAAC;AAED,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,gBAAgB,EAAE,QAAQ,6EAA6E,EAAE,QAAQ,CAAC,SAAS,KAAK,eAAe,oBAAoB,EAAE,SAAS,KAAK,OAAO,SAAS,OAAO,EAAE,SAAS,OAAO,UAAU;AAC9Q,WAAK,OAAO,SAAS,UAAU,MAAM,KAAK;AAC1C,YAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IACrC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,cAAc,EAAE,QAAQ,cAAc,EAAE,UAAU,CAAC,WAAW,OAAO,cAAc,cAAc,EAAE,QAAQ,YAAY;AAE/J,YAAM,OAAO,MAAM,KAAK,OAAO,aAAa;AAC5C,UAAG,MAAM;AACP,YAAI,SAAS,OAAO,qCAAqC;AAAA,MAC3D,OAAK;AACH,YAAI,SAAS,OAAO,wDAAwD;AAAA,MAC9E;AAAA,IACF,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,kBAAkB,EAAE,QAAQ,wCAAwC,EAAE,YAAY,CAAC,aAAa;AACxI,eAAS,UAAU,qBAAqB,mBAAmB;AAC3D,eAAS,UAAU,SAAS,4BAA4B;AACxD,eAAS,UAAU,iBAAiB,oBAAoB;AACxD,eAAS,UAAU,sBAAsB,oBAAoB;AAC7D,eAAS,SAAS,OAAO,UAAU;AACjC,aAAK,OAAO,SAAS,mBAAmB;AACxC,cAAM,KAAK,OAAO,aAAa;AAAA,MACjC,CAAC;AACD,eAAS,SAAS,KAAK,OAAO,SAAS,gBAAgB;AAAA,IACzD,CAAC;AAED,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,kBAAkB,EAAE,QAAQ,oHAAoH,EAAE,YAAY,CAAC,aAAa;AAEpN,YAAM,YAAY,OAAO,KAAK,iBAAiB;AAC/C,eAAQ,IAAI,GAAG,IAAI,UAAU,QAAQ,KAAK;AACxC,iBAAS,UAAU,UAAU,CAAC,GAAG,UAAU,CAAC,CAAC;AAAA,MAC/C;AACA,eAAS,SAAS,OAAO,UAAU;AACjC,aAAK,OAAO,SAAS,WAAW;AAChC,cAAM,KAAK,OAAO,aAAa;AAC/B,+BAAuB,QAAQ,KAAK,kBAAkB,CAAC;AAEvD,cAAM,YAAY,KAAK,IAAI,UAAU,gBAAgB,gCAAgC,EAAE,SAAS,IAAI,KAAK,IAAI,UAAU,gBAAgB,gCAAgC,EAAE,CAAC,EAAE,OAAO;AACnL,YAAG,WAAW;AACZ,oBAAU,SAAS;AAAA,QACrB;AAAA,MACF,CAAC;AACD,eAAS,SAAS,KAAK,OAAO,SAAS,QAAQ;AAAA,IACjD,CAAC;AAED,UAAM,yBAAyB,YAAY,SAAS,QAAQ;AAAA,MAC1D,MAAM,KAAK,kBAAkB;AAAA,IAC/B,CAAC;AACD,gBAAY,SAAS,MAAM;AAAA,MACzB,MAAM;AAAA,IACR,CAAC;AAED,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,iBAAiB,EAAE,QAAQ,gDAAgD,EAAE,QAAQ,CAAC,SAAS,KAAK,eAAe,uBAAuB,EAAE,SAAS,KAAK,OAAO,SAAS,eAAe,EAAE,SAAS,OAAO,UAAU;AAC7P,WAAK,OAAO,SAAS,kBAAkB;AACvC,YAAM,KAAK,OAAO,aAAa;AAAA,IACjC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,mBAAmB,EAAE,QAAQ,kDAAkD,EAAE,QAAQ,CAAC,SAAS,KAAK,eAAe,uBAAuB,EAAE,SAAS,KAAK,OAAO,SAAS,iBAAiB,EAAE,SAAS,OAAO,UAAU;AACnQ,WAAK,OAAO,SAAS,oBAAoB;AACzC,YAAM,KAAK,OAAO,aAAa;AAAA,IACjC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,WAAW,EAAE,QAAQ,4CAA4C,EAAE,QAAQ,CAAC,SAAS,KAAK,eAAe,uBAAuB,EAAE,SAAS,KAAK,OAAO,SAAS,SAAS,EAAE,SAAS,OAAO,UAAU;AAC7O,WAAK,OAAO,SAAS,YAAY;AACjC,YAAM,KAAK,OAAO,aAAa;AAAA,IACjC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,mBAAmB,EAAE,QAAQ,2EAA2E,EAAE,QAAQ,CAAC,SAAS,KAAK,eAAe,uBAAuB,EAAE,SAAS,KAAK,OAAO,SAAS,iBAAiB,EAAE,SAAS,OAAO,UAAU;AAC5R,WAAK,OAAO,SAAS,oBAAoB;AACzC,YAAM,KAAK,OAAO,aAAa;AAAA,IACjC,CAAC,CAAC;AACF,gBAAY,SAAS,MAAM;AAAA,MACzB,MAAM;AAAA,IACR,CAAC;AAED,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,gBAAgB,EAAE,QAAQ,yBAAyB,EAAE,UAAU,CAAC,WAAW,OAAO,SAAS,KAAK,OAAO,SAAS,cAAc,EAAE,SAAS,OAAO,UAAU;AAClM,WAAK,OAAO,SAAS,iBAAiB;AACtC,YAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IACrC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,eAAe,EAAE,QAAQ,2BAA2B,EAAE,UAAU,CAAC,WAAW,OAAO,SAAS,KAAK,OAAO,SAAS,aAAa,EAAE,SAAS,OAAO,UAAU;AAClM,WAAK,OAAO,SAAS,gBAAgB;AACrC,YAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IACrC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,uBAAuB,EAAE,QAAQ,wBAAwB,EAAE,UAAU,CAAC,WAAW,OAAO,SAAS,KAAK,OAAO,SAAS,qBAAqB,EAAE,SAAS,OAAO,UAAU;AAC/M,WAAK,OAAO,SAAS,wBAAwB;AAC7C,YAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IACrC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,WAAW,EAAE,QAAQ,gCAAgC,EAAE,UAAU,CAAC,WAAW,OAAO,SAAS,KAAK,OAAO,SAAS,SAAS,EAAE,SAAS,OAAO,UAAU;AAC/L,WAAK,OAAO,SAAS,YAAY;AACjC,YAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IACrC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,WAAW,EAAE,QAAQ,gCAAgC,EAAE,UAAU,CAAC,WAAW,OAAO,SAAS,KAAK,OAAO,SAAS,SAAS,EAAE,SAAS,OAAO,UAAU;AAC/L,WAAK,OAAO,SAAS,YAAY;AACjC,YAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IACrC,CAAC,CAAC;AACF,gBAAY,SAAS,MAAM;AAAA,MACzB,MAAM;AAAA,IACR,CAAC;AAED,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,YAAY,EAAE,QAAQ,uDAAuD,EAAE,UAAU,CAAC,WAAW,OAAO,SAAS,KAAK,OAAO,SAAS,UAAU,EAAE,SAAS,OAAO,UAAU;AACxN,WAAK,OAAO,SAAS,aAAa;AAClC,YAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IACrC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,kBAAkB,EAAE,QAAQ,6DAA6D,EAAE,UAAU,CAAC,WAAW,OAAO,SAAS,KAAK,OAAO,SAAS,gBAAgB,EAAE,SAAS,OAAO,UAAU;AAC1O,WAAK,OAAO,SAAS,mBAAmB;AACxC,YAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IACrC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,eAAe,EAAE,QAAQ,0KAA0K,EAAE,UAAU,CAAC,WAAW,OAAO,SAAS,KAAK,OAAO,SAAS,aAAa,EAAE,SAAS,OAAO,UAAU;AACjV,WAAK,OAAO,SAAS,gBAAgB;AACrC,YAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IACrC,CAAC,CAAC;AAEF,gBAAY,SAAS,MAAM;AAAA,MACzB,MAAM;AAAA,IACR,CAAC;AAED,gBAAY,SAAS,MAAM;AAAA,MACzB,MAAM;AAAA,IACR,CAAC;AACD,QAAI,sBAAsB,YAAY,SAAS,KAAK;AACpD,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,aAAa,EAAE,QAAQ,yBAAyB,EAAE,UAAU,CAAC,WAAW,OAAO,cAAc,aAAa,EAAE,QAAQ,YAAY;AAExK,UAAI,QAAQ,wDAAwD,GAAG;AAErE,YAAG;AACD,gBAAM,KAAK,OAAO,wBAAwB,IAAI;AAC9C,8BAAoB,YAAY;AAAA,QAClC,SAAO,GAAN;AACC,8BAAoB,YAAY,uCAAuC;AAAA,QACzE;AAAA,MACF;AAAA,IACF,CAAC,CAAC;AAGF,gBAAY,SAAS,MAAM;AAAA,MACzB,MAAM;AAAA,IACR,CAAC;AACD,QAAI,cAAc,YAAY,SAAS,KAAK;AAC5C,SAAK,uBAAuB,WAAW;AAGvC,gBAAY,SAAS,MAAM;AAAA,MACzB,MAAM;AAAA,IACR,CAAC;AACD,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,eAAe,EAAE,QAAQ,oKAAoK,EAAE,UAAU,CAAC,WAAW,OAAO,cAAc,eAAe,EAAE,QAAQ,YAAY;AAEvT,UAAI,QAAQ,0HAA0H,GAAG;AAEvI,cAAM,KAAK,OAAO,8BAA8B;AAAA,MAClD;AAAA,IACF,CAAC,CAAC;AAAA,EAEJ;AAAA,EACA,oBAAoB;AAClB,WAAO,cAAc,kBAAkB,KAAK,OAAO,SAAS,QAAQ,EAAE,QAAQ,KAAK,IAAI;AAAA,EACzF;AAAA,EAEA,uBAAuB,aAAa;AAClC,gBAAY,MAAM;AAClB,QAAG,KAAK,OAAO,SAAS,aAAa,SAAS,GAAG;AAE/C,kBAAY,SAAS,KAAK;AAAA,QACxB,MAAM;AAAA,MACR,CAAC;AACD,UAAI,OAAO,YAAY,SAAS,IAAI;AACpC,eAAS,eAAe,KAAK,OAAO,SAAS,cAAc;AACzD,aAAK,SAAS,MAAM;AAAA,UAClB,MAAM;AAAA,QACR,CAAC;AAAA,MACH;AAEA,UAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,oBAAoB,EAAE,QAAQ,yBAAyB,EAAE,UAAU,CAAC,WAAW,OAAO,cAAc,yBAAyB,EAAE,QAAQ,YAAY;AAE3L,oBAAY,MAAM;AAElB,oBAAY,SAAS,KAAK;AAAA,UACxB,MAAM;AAAA,QACR,CAAC;AACD,cAAM,KAAK,OAAO,mBAAmB;AAErC,aAAK,uBAAuB,WAAW;AAAA,MACzC,CAAC,CAAC;AAAA,IACJ,OAAK;AACH,kBAAY,SAAS,KAAK;AAAA,QACxB,MAAM;AAAA,MACR,CAAC;AAAA,IACH;AAAA,EACF;AACF;AAEA,SAAS,gBAAgB,MAAM;AAC7B,SAAQ,KAAK,QAAQ,GAAG,MAAM,KAAO,CAAC,KAAK,GAAG,EAAE,QAAQ,KAAK,CAAC,CAAC,MAAM;AACvE;AAEA,IAAM,mCAAmC;AAEzC,IAAM,2BAAN,cAAuC,SAAS,SAAS;AAAA,EACvD,YAAY,MAAM,QAAQ;AACxB,UAAM,IAAI;AACV,SAAK,SAAS;AACd,SAAK,aAAa;AAClB,SAAK,gBAAgB;AACrB,SAAK,cAAc;AACnB,SAAK,OAAO;AACZ,SAAK,WAAW;AAChB,SAAK,iBAAiB;AACtB,SAAK,kBAAkB,CAAC;AACxB,SAAK,QAAQ,CAAC;AACd,SAAK,YAAY;AACjB,SAAK,oBAAoB;AACzB,SAAK,gBAAgB;AAAA,EACvB;AAAA,EACA,iBAAiB;AACf,WAAO;AAAA,EACT;AAAA,EACA,UAAU;AACR,WAAO;AAAA,EACT;AAAA,EACA,cAAc;AACZ,WAAO;AAAA,EACT;AAAA,EACA,SAAS;AACP,SAAK,SAAS;AACd,SAAK,OAAO,gBAAgB;AAAA,EAC9B;AAAA,EACA,UAAU;AACR,SAAK,KAAK,UAAU;AACpB,SAAK,IAAI,UAAU,0BAA0B,gCAAgC;AAAA,EAC/E;AAAA,EACA,cAAc;AACZ,SAAK,YAAY,MAAM;AACvB,SAAK,iBAAiB,KAAK,YAAY,UAAU,mBAAmB;AAEpE,SAAK,eAAe;AAEpB,SAAK,gBAAgB;AAErB,SAAK,kBAAkB;AACvB,SAAK,OAAO,aAAa,KAAK,aAAa,MAAM;AAAA,EACnD;AAAA;AAAA,EAEA,iBAAiB;AAEf,QAAI,oBAAoB,KAAK,eAAe,UAAU,sBAAsB;AAE5E,QAAI,YAAW,KAAK,KAAK,KAAK;AAC9B,QAAI,kBAAkB,kBAAkB,SAAS,SAAS;AAAA,MACxD,MAAM;AAAA,QACJ,MAAM;AAAA,QACN,OAAO;AAAA,MACT;AAAA,MACA,KAAK;AAAA,IACP,CAAC;AACD,oBAAgB,iBAAiB,UAAU,KAAK,YAAY,KAAK,IAAI,CAAC;AAGtE,QAAI,iBAAiB,KAAK,sBAAsB,mBAAmB,cAAc,mBAAmB;AACpG,mBAAe,iBAAiB,SAAS,KAAK,gBAAgB,KAAK,IAAI,CAAC;AAExE,QAAI,WAAW,KAAK,sBAAsB,mBAAmB,aAAa,MAAM;AAChF,aAAS,iBAAiB,SAAS,KAAK,UAAU,KAAK,IAAI,CAAC;AAE5D,QAAI,cAAc,KAAK,sBAAsB,mBAAmB,gBAAgB,SAAS;AACzF,gBAAY,iBAAiB,SAAS,KAAK,kBAAkB,KAAK,IAAI,CAAC;AAEvE,UAAM,eAAe,KAAK,sBAAsB,mBAAmB,YAAY,MAAM;AACrF,iBAAa,iBAAiB,SAAS,KAAK,SAAS,KAAK,IAAI,CAAC;AAAA,EACjE;AAAA,EACA,MAAM,oBAAoB;AACxB,UAAM,SAAS,MAAM,KAAK,IAAI,MAAM,QAAQ,KAAK,0BAA0B;AAC3E,SAAK,QAAQ,OAAO,MAAM,IAAI,CAAC,SAAS;AACtC,aAAO,KAAK,QAAQ,6BAA6B,EAAE,EAAE,QAAQ,SAAS,EAAE;AAAA,IAC1E,CAAC;AAED,QAAI,CAAC,KAAK;AACR,WAAK,QAAQ,IAAI,iCAAiC,KAAK,KAAK,IAAI;AAClE,SAAK,MAAM,KAAK;AAAA,EAClB;AAAA,EAEA,sBAAsB,mBAAmB,OAAO,OAAK,MAAM;AACzD,QAAI,MAAM,kBAAkB,SAAS,UAAU;AAAA,MAC7C,MAAM;AAAA,QACJ;AAAA,MACF;AAAA,IACF,CAAC;AACD,QAAG,MAAK;AACN,eAAS,QAAQ,KAAK,IAAI;AAAA,IAC5B,OAAK;AACH,UAAI,YAAY;AAAA,IAClB;AACA,WAAO;AAAA,EACT;AAAA;AAAA,EAEA,WAAW;AACT,SAAK,WAAW;AAChB,SAAK,YAAY;AAEjB,SAAK,oBAAoB,WAAW;AACpC,SAAK,WAAW,YAAY,QAAQ,kBAAkB,KAAK,OAAO,SAAS,QAAQ,EAAE,kBAAgB;AAAA,EACvG;AAAA;AAAA,EAEA,MAAM,UAAU,SAAS;AACvB,SAAK,WAAW;AAChB,UAAM,KAAK,KAAK,UAAU,OAAO;AACjC,SAAK,YAAY;AACjB,aAAS,IAAI,GAAG,IAAI,KAAK,KAAK,QAAQ,QAAQ,KAAK;AACjD,YAAM,KAAK,eAAe,KAAK,KAAK,QAAQ,CAAC,EAAE,SAAS,KAAK,KAAK,QAAQ,CAAC,EAAE,IAAI;AAAA,IACnF;AAAA,EACF;AAAA;AAAA,EAEA,aAAa;AACX,QAAI,KAAK,MAAM;AACb,WAAK,KAAK,UAAU;AAAA,IACtB;AACA,SAAK,OAAO,IAAI,0BAA0B,KAAK,MAAM;AAErD,QAAI,KAAK,oBAAoB;AAC3B,oBAAc,KAAK,kBAAkB;AAAA,IACvC;AAEA,SAAK,kBAAkB,CAAC;AAExB,SAAK,WAAW;AAAA,EAClB;AAAA,EAEA,YAAY,OAAO;AACjB,QAAI,gBAAgB,MAAM,OAAO;AACjC,SAAK,KAAK,YAAY,aAAa;AAAA,EACrC;AAAA;AAAA,EAGA,YAAY;AACV,SAAK,KAAK,UAAU;AACpB,QAAI,SAAS,OAAO,gCAAgC;AAAA,EACtD;AAAA,EAEA,kBAAkB;AAChB,SAAK,OAAO,UAAU;AAAA,EACxB;AAAA;AAAA,EAEA,kBAAkB;AAEhB,SAAK,WAAW,KAAK,eAAe,UAAU,aAAa;AAE3D,SAAK,oBAAoB,KAAK,SAAS,UAAU,sBAAsB;AAAA,EACzE;AAAA;AAAA,EAEA,6BAA6B;AAE3B,QAAG,CAAC,KAAK;AAAe,WAAK,gBAAgB,IAAI,gCAAgC,KAAK,KAAK,IAAI;AAC/F,SAAK,cAAc,KAAK;AAAA,EAC1B;AAAA;AAAA,EAEA,MAAM,+BAA+B;AAEnC,QAAG,CAAC,KAAK,iBAAgB;AACvB,WAAK,kBAAkB,IAAI,kCAAkC,KAAK,KAAK,IAAI;AAAA,IAC7E;AACA,SAAK,gBAAgB,KAAK;AAAA,EAC5B;AAAA;AAAA,EAEA,iBAAiB,aAAa;AAE5B,QAAI,YAAY,KAAK,SAAS;AAE9B,QAAI,cAAc,KAAK,SAAS,MAAM,UAAU,GAAG,SAAS;AAE5D,QAAI,aAAa,KAAK,SAAS,MAAM,UAAU,WAAW,KAAK,SAAS,MAAM,MAAM;AAEpF,SAAK,SAAS,QAAQ,cAAc,cAAc;AAElD,SAAK,SAAS,iBAAiB,YAAY,YAAY;AACvD,SAAK,SAAS,eAAe,YAAY,YAAY;AAErD,SAAK,SAAS,MAAM;AAAA,EACtB;AAAA;AAAA,EAGA,oBAAoB;AAElB,QAAI,aAAa,KAAK,eAAe,UAAU,cAAc;AAE7D,SAAK,WAAW,WAAW,SAAS,YAAY;AAAA,MAC9C,KAAK;AAAA,MACL,MAAM;AAAA,QACJ,aAAa;AAAA,MACf;AAAA,IACF,CAAC;AAID,eAAW,iBAAiB,SAAS,CAAC,MAAM;AAC1C,UAAG,CAAC,KAAK,GAAG,EAAE,QAAQ,EAAE,GAAG,MAAM;AAAI;AACrC,YAAM,YAAY,KAAK,SAAS;AAEhC,UAAI,EAAE,QAAQ,KAAK;AAEjB,YAAG,KAAK,SAAS,MAAM,YAAY,CAAC,MAAM,KAAI;AAE5C,eAAK,2BAA2B;AAChC;AAAA,QACF;AAAA,MACF,OAAK;AACH,aAAK,cAAc;AAAA,MACrB;AAEA,UAAI,EAAE,QAAQ,KAAK;AAGjB,YAAI,KAAK,SAAS,MAAM,WAAW,KAAK,KAAK,SAAS,MAAM,YAAY,CAAC,MAAM,KAAK;AAElF,eAAK,6BAA6B;AAClC;AAAA,QACF;AAAA,MACF;AAAA,IAEF,CAAC;AAED,eAAW,iBAAiB,WAAW,CAAC,MAAM;AAC5C,UAAI,EAAE,QAAQ,WAAW,EAAE,UAAU;AACnC,UAAE,eAAe;AACjB,YAAG,KAAK,eAAc;AACpB,kBAAQ,IAAI,yCAAyC;AACrD,cAAI,SAAS,OAAO,6DAA6D;AACjF;AAAA,QACF;AAEA,YAAI,aAAa,KAAK,SAAS;AAE/B,aAAK,SAAS,QAAQ;AAEtB,aAAK,oBAAoB,UAAU;AAAA,MACrC;AACA,WAAK,SAAS,MAAM,SAAS;AAC7B,WAAK,SAAS,MAAM,SAAU,KAAK,SAAS,eAAgB;AAAA,IAC9D,CAAC;AAED,QAAI,mBAAmB,WAAW,UAAU,qBAAqB;AAEjE,QAAI,eAAe,iBAAiB,SAAS,QAAQ,EAAE,MAAM,EAAC,IAAI,mBAAmB,OAAO,iBAAgB,EAAE,CAAC;AAC/G,aAAS,QAAQ,cAAc,QAAQ;AAEvC,iBAAa,iBAAiB,SAAS,MAAM;AAE3C,WAAK,WAAW;AAAA,IAClB,CAAC;AAED,QAAI,SAAS,iBAAiB,SAAS,UAAU,EAAE,MAAM,EAAC,IAAI,iBAAgB,GAAG,KAAK,cAAc,CAAC;AACrG,WAAO,YAAY;AAEnB,WAAO,iBAAiB,SAAS,MAAM;AACrC,UAAG,KAAK,eAAc;AACpB,gBAAQ,IAAI,yCAAyC;AACrD,YAAI,SAAS,OAAO,yCAAyC;AAC7D;AAAA,MACF;AAEA,UAAI,aAAa,KAAK,SAAS;AAE/B,WAAK,SAAS,QAAQ;AAEtB,WAAK,oBAAoB,UAAU;AAAA,IACrC,CAAC;AAAA,EACH;AAAA,EACA,MAAM,oBAAoB,YAAY;AACpC,SAAK,iBAAiB;AAEtB,UAAM,KAAK,eAAe,YAAY,MAAM;AAC5C,SAAK,KAAK,sBAAsB;AAAA,MAC9B,MAAM;AAAA,MACN,SAAS;AAAA,IACX,CAAC;AACD,UAAM,KAAK,iBAAiB;AAG5B,QAAG,KAAK,KAAK,uBAAuB,UAAU,GAAG;AAC/C,WAAK,KAAK,+BAA+B,YAAY,IAAI;AACzD;AAAA,IACF;AAQA,QAAG,KAAK,mCAAmC,UAAU,KAAK,KAAK,KAAK,0BAA0B,UAAU,GAAG;AAEzG,YAAM,UAAU,MAAM,KAAK,iBAAiB,UAAU;AAItD,YAAM,SAAS;AAAA,QACb;AAAA,UACE,MAAM;AAAA;AAAA,UAEN,SAAS;AAAA,QACX;AAAA,QACA;AAAA,UACE,MAAM;AAAA,UACN,SAAS;AAAA,QACX;AAAA,MACF;AACA,WAAK,2BAA2B,EAAC,UAAU,QAAQ,aAAa,EAAC,CAAC;AAClE;AAAA,IACF;AAEA,SAAK,2BAA2B;AAAA,EAClC;AAAA,EAEA,MAAM,mBAAmB;AACvB,QAAI,KAAK;AACP,oBAAc,KAAK,kBAAkB;AACvC,UAAM,KAAK,eAAe,OAAO,WAAW;AAE5C,QAAI,OAAO;AACX,SAAK,WAAW,YAAY;AAC5B,SAAK,qBAAqB,YAAY,MAAM;AAC1C;AACA,UAAI,OAAO;AACT,eAAO;AACT,WAAK,WAAW,YAAY,IAAI,OAAO,IAAI;AAAA,IAC7C,GAAG,GAAG;AAAA,EAGR;AAAA,EAEA,mBAAmB;AACjB,SAAK,gBAAgB;AAErB,QAAG,SAAS,eAAe,gBAAgB;AACzC,eAAS,eAAe,gBAAgB,EAAE,MAAM,UAAU;AAE5D,QAAG,SAAS,eAAe,iBAAiB;AAC1C,eAAS,eAAe,iBAAiB,EAAE,MAAM,UAAU;AAAA,EAC/D;AAAA,EACA,qBAAqB;AACnB,SAAK,gBAAgB;AAErB,QAAG,SAAS,eAAe,gBAAgB;AACzC,eAAS,eAAe,gBAAgB,EAAE,MAAM,UAAU;AAE5D,QAAG,SAAS,eAAe,iBAAiB;AAC1C,eAAS,eAAe,iBAAiB,EAAE,MAAM,UAAU;AAAA,EAC/D;AAAA;AAAA,EAIA,mCAAmC,YAAY;AAC7C,UAAM,UAAU,WAAW,MAAM,KAAK,OAAO,iBAAiB;AAC9D,QAAG;AAAS,aAAO;AACnB,WAAO;AAAA,EACT;AAAA;AAAA,EAGA,MAAM,eAAe,SAAS,OAAK,aAAa,cAAY,OAAO;AAEjE,QAAG,KAAK,oBAAoB;AAC1B,oBAAc,KAAK,kBAAkB;AACrC,WAAK,qBAAqB;AAE1B,WAAK,WAAW,YAAY;AAAA,IAC9B;AACA,QAAG,aAAa;AACd,WAAK,uBAAuB;AAC5B,UAAG,QAAQ,QAAQ,IAAI,MAAM,IAAI;AAC/B,aAAK,WAAW,aAAa;AAAA,MAC/B,OAAK;AACH,aAAK,WAAW,YAAY;AAE5B,cAAM,SAAS,iBAAiB,eAAe,KAAK,qBAAqB,KAAK,YAAY,gBAAgB,IAAI,SAAS,UAAU,CAAC;AAAA,MACpI;AAAA,IACF,OAAK;AACH,WAAK,sBAAsB;AAC3B,UAAI,KAAK,KAAK,OAAO,WAAW,KAAO,KAAK,cAAc,MAAO;AAE/D,aAAK,oBAAoB,IAAI;AAAA,MAC/B;AAEA,WAAK,WAAW,YAAY;AAC5B,YAAM,SAAS,iBAAiB,eAAe,SAAS,KAAK,YAAY,gBAAgB,IAAI,SAAS,UAAU,CAAC;AAEjH,WAAK,wBAAwB;AAE7B,WAAK,8BAA8B,OAAO;AAAA,IAC5C;AAEA,SAAK,kBAAkB,YAAY,KAAK,kBAAkB;AAAA,EAC5D;AAAA,EACA,8BAA8B,SAAS;AACrC,QAAI,KAAK,KAAK,WAAW,KAAK,KAAK,KAAK;AAEtC,YAAM,eAAe,KAAK,WAAW,SAAS,QAAQ;AAAA,QACpD,KAAK;AAAA,QACL,MAAM;AAAA,UACJ,OAAO;AAAA;AAAA,QACT;AAAA,MACF,CAAC;AACD,YAAM,WAAW,KAAK,KAAK;AAC3B,eAAS,QAAQ,cAAc,KAAK;AACpC,mBAAa,iBAAiB,SAAS,MAAM;AAE3C,kBAAU,UAAU,UAAU,2BAA2B,WAAW,SAAS;AAC7E,YAAI,SAAS,OAAO,4DAA4D;AAAA,MAClF,CAAC;AAAA,IACH;AACA,QAAG,KAAK,KAAK,SAAS;AAEpB,YAAM,qBAAqB,KAAK,WAAW,SAAS,QAAQ;AAAA,QAC1D,KAAK;AAAA,QACL,MAAM;AAAA,UACJ,OAAO;AAAA;AAAA,QACT;AAAA,MACF,CAAC;AACD,YAAM,eAAe,KAAK,KAAK,QAAQ,QAAQ,WAAW,MAAO,EAAE,SAAS;AAC5E,eAAS,QAAQ,oBAAoB,OAAO;AAC5C,yBAAmB,iBAAiB,SAAS,MAAM;AAEjD,kBAAU,UAAU,UAAU,wBAAwB,eAAe,SAAS;AAC9E,YAAI,SAAS,OAAO,iDAAiD;AAAA,MACvE,CAAC;AAAA,IACH;AAEA,UAAM,cAAc,KAAK,WAAW,SAAS,QAAQ;AAAA,MACnD,KAAK;AAAA,MACL,MAAM;AAAA,QACJ,OAAO;AAAA;AAAA,MACT;AAAA,IACF,CAAC;AACD,aAAS,QAAQ,aAAa,MAAM;AACpC,gBAAY,iBAAiB,SAAS,MAAM;AAE1C,gBAAU,UAAU,UAAU,QAAQ,SAAS,CAAC;AAChD,UAAI,SAAS,OAAO,iDAAiD;AAAA,IACvE,CAAC;AAAA,EACH;AAAA,EAEA,0BAA0B;AACxB,UAAM,QAAQ,KAAK,WAAW,iBAAiB,GAAG;AAElD,QAAI,MAAM,SAAS,GAAG;AACpB,eAAS,IAAI,GAAG,IAAI,MAAM,QAAQ,KAAK;AACrC,cAAM,OAAO,MAAM,CAAC;AACpB,cAAM,YAAY,KAAK,aAAa,WAAW;AAE/C,aAAK,iBAAiB,aAAa,CAAC,UAAU;AAC5C,eAAK,IAAI,UAAU,QAAQ,cAAc;AAAA,YACvC;AAAA,YACA,QAAQ;AAAA,YACR,aAAa,KAAK;AAAA,YAClB,UAAU;AAAA;AAAA,YAEV,UAAU;AAAA,UACZ,CAAC;AAAA,QACH,CAAC;AAED,aAAK,iBAAiB,SAAS,CAAC,UAAU;AACxC,gBAAM,aAAa,KAAK,IAAI,cAAc,qBAAqB,WAAW,GAAG;AAE7E,gBAAM,MAAM,SAAS,OAAO,WAAW,KAAK;AAE5C,cAAI,OAAO,KAAK,IAAI,UAAU,QAAQ,GAAG;AACzC,eAAK,SAAS,UAAU;AAAA,QAC1B,CAAC;AAAA,MACH;AAAA,IACF;AAAA,EACF;AAAA,EAEA,oBAAoB,MAAM;AACxB,QAAI,aAAa,KAAK,kBAAkB,UAAU,cAAc,MAAM;AAEtE,SAAK,aAAa,WAAW,UAAU,oBAAoB;AAE3D,SAAK,YAAY;AAAA,EACnB;AAAA,EAEA,MAAM,2BAA2B,OAAK,CAAC,GAAG;AACxC,UAAM,UAAU,KAAK,YAAY,KAAK,WAAW,KAAK,KAAK,gBAAgB;AAC3E,YAAQ,IAAI,WAAW,OAAO;AAC9B,UAAM,mBAAmB,KAAK,MAAM,cAAc,KAAK,OAAO,SAAS,gBAAgB,IAAI,CAAC;AAC5F,YAAQ,IAAI,oBAAoB,gBAAgB;AAChD,UAAM,iBAAiB,KAAK,MAAM,KAAK,UAAU,OAAO,EAAE,SAAS,CAAC;AACpE,YAAQ,IAAI,kBAAkB,cAAc;AAC5C,QAAI,uBAAuB,mBAAmB;AAE9C,QAAG,uBAAuB;AAAG,6BAAuB;AAAA,aAC5C,uBAAuB;AAAM,6BAAuB;AAC5D,YAAQ,IAAI,wBAAwB,oBAAoB;AACxD,WAAO;AAAA,MACL,OAAO,KAAK,OAAO,SAAS;AAAA,MAC5B,UAAU;AAAA;AAAA,MAEV,YAAY;AAAA,MACZ,aAAa;AAAA,MACb,OAAO;AAAA,MACP,kBAAkB;AAAA,MAClB,mBAAmB;AAAA,MACnB,QAAQ;AAAA,MACR,MAAM;AAAA,MACN,GAAG;AAAA;AAAA,MAEH,GAAG;AAAA,IACL;AAEA,QAAG,KAAK,QAAQ;AACd,YAAM,WAAW,MAAM,IAAI,QAAQ,CAAC,SAAS,WAAW;AACtD,YAAI;AAEF,gBAAM,MAAM;AACZ,eAAK,gBAAgB,IAAI,WAAW,KAAK;AAAA,YACvC,SAAS;AAAA,cACP,gBAAgB;AAAA,cAChB,eAAe,UAAU,KAAK,OAAO,SAAS;AAAA,YAChD;AAAA,YACA,QAAQ;AAAA,YACR,SAAS,KAAK,UAAU,IAAI;AAAA,UAC9B,CAAC;AACD,cAAI,MAAM;AACV,eAAK,cAAc,iBAAiB,WAAW,CAAC,MAAM;AACpD,gBAAI,EAAE,QAAQ,UAAU;AACtB,oBAAM,UAAU,KAAK,MAAM,EAAE,IAAI;AACjC,oBAAM,OAAO,QAAQ,QAAQ,CAAC,EAAE,MAAM;AACtC,kBAAI,CAAC,MAAM;AACT;AAAA,cACF;AACA,qBAAO;AACP,mBAAK,eAAe,MAAM,aAAa,IAAI;AAAA,YAC7C,OAAO;AACL,mBAAK,WAAW;AAChB,sBAAQ,GAAG;AAAA,YACb;AAAA,UACF,CAAC;AACD,eAAK,cAAc,iBAAiB,oBAAoB,CAAC,MAAM;AAC7D,gBAAI,EAAE,cAAc,GAAG;AACrB,sBAAQ,IAAI,iBAAiB,EAAE,UAAU;AAAA,YAC3C;AAAA,UACF,CAAC;AACD,eAAK,cAAc,iBAAiB,SAAS,CAAC,MAAM;AAClD,oBAAQ,MAAM,CAAC;AACf,gBAAI,SAAS,OAAO,sEAAsE;AAC1F,iBAAK,eAAe,8CAA8C,WAAW;AAC7E,iBAAK,WAAW;AAChB,mBAAO,CAAC;AAAA,UACV,CAAC;AACD,eAAK,cAAc,OAAO;AAAA,QAC5B,SAAS,KAAP;AACA,kBAAQ,MAAM,GAAG;AACjB,cAAI,SAAS,OAAO,sEAAsE;AAC1F,eAAK,WAAW;AAChB,iBAAO,GAAG;AAAA,QACZ;AAAA,MACF,CAAC;AAED,YAAM,KAAK,eAAe,UAAU,WAAW;AAC/C,WAAK,KAAK,sBAAsB;AAAA,QAC9B,MAAM;AAAA,QACN,SAAS;AAAA,MACX,CAAC;AACD;AAAA,IACF,OAAK;AACH,UAAG;AACD,cAAM,WAAW,OAAO,GAAG,SAAS,YAAY;AAAA,UAC9C,KAAK;AAAA,UACL,QAAQ;AAAA,UACR,SAAS;AAAA,YACP,eAAe,UAAU,KAAK,OAAO,SAAS;AAAA,YAC9C,gBAAgB;AAAA,UAClB;AAAA,UACA,aAAa;AAAA,UACb,MAAM,KAAK,UAAU,IAAI;AAAA,UACzB,OAAO;AAAA,QACT,CAAC;AAED,eAAO,KAAK,MAAM,SAAS,IAAI,EAAE,QAAQ,CAAC,EAAE,QAAQ;AAAA,MACtD,SAAO,KAAN;AACC,YAAI,SAAS,OAAO,kCAAkC,KAAK;AAAA,MAC7D;AAAA,IACF;AAAA,EACF;AAAA,EAEA,aAAa;AACX,QAAG,KAAK,eAAc;AACpB,WAAK,cAAc,MAAM;AACzB,WAAK,gBAAgB;AAAA,IACvB;AACA,SAAK,mBAAmB;AACxB,QAAG,KAAK,oBAAmB;AACzB,oBAAc,KAAK,kBAAkB;AACrC,WAAK,qBAAqB;AAE1B,WAAK,WAAW,cAAc,OAAO;AACrC,WAAK,aAAa;AAAA,IACpB;AAAA,EACF;AAAA,EAEA,MAAM,iBAAiB,YAAY;AACjC,SAAK,KAAK,cAAc;AAExB,UAAM,YAAY;AAElB,UAAM,SAAS;AAAA,MACb;AAAA,QACE,MAAM;AAAA,QACN,SAAS;AAAA,MACX;AAAA,MACA;AAAA,QACE,MAAM;AAAA,QACN,SAAS;AAAA,MACX;AAAA,IACF;AACA,UAAM,MAAM,MAAM,KAAK,2BAA2B;AAAA,MAChD,UAAU;AAAA,MACV,QAAQ;AAAA,MACR,aAAa;AAAA,MACb,YAAY;AAAA,IACd,CAAC;AACD,SAAK,KAAK,MAAM;AAEhB,QAAI,SAAS,CAAC;AAEd,QAAG,KAAK,KAAK,0BAA0B,UAAU,GAAG;AAElD,YAAM,cAAc,KAAK,KAAK,sBAAsB,UAAU;AAG9D,UAAG,aAAY;AACb,iBAAS;AAAA,UACP,kBAAkB;AAAA,QACpB;AAAA,MACF;AAAA,IACF;AAEA,QAAI,UAAU,MAAM,KAAK,OAAO,IAAI,OAAO,KAAK,MAAM;AACtD,YAAQ,IAAI,WAAW,QAAQ,MAAM;AACrC,cAAU,KAAK,2CAA2C,OAAO;AACjE,YAAQ,IAAI,+BAA+B,QAAQ,MAAM;AACzD,cAAU,KAAK,gCAAgC,OAAO;AAEtD,WAAO,MAAM,KAAK,uBAAuB,OAAO;AAAA,EAClD;AAAA,EAGA,gCAAgC,SAAS;AAEvC,cAAU,QAAQ,KAAK,CAAC,GAAG,MAAM;AAC/B,YAAM,UAAU,EAAE,aAAa,EAAE;AACjC,YAAM,UAAU,EAAE,aAAa,EAAE;AAEjC,UAAI,UAAU;AACZ,eAAO;AAET,UAAI,UAAU;AACZ,eAAO;AAET,aAAO;AAAA,IACT,CAAC;AACD,WAAO;AAAA,EACT;AAAA,EAEA,2CAA2C,SAAS;AAElD,UAAM,MAAM,QAAQ,IAAI,CAAC,MAAM,EAAE,UAAU;AAC3C,UAAM,OAAO,IAAI,OAAO,CAAC,GAAG,MAAM,IAAI,CAAC,IAAI,IAAI;AAC/C,QAAI,UAAU,KAAK,KAAK,IAAI,IAAI,CAAC,MAAM,KAAK,IAAI,IAAI,MAAM,CAAC,CAAC,EAAE,OAAO,CAAC,GAAG,MAAM,IAAI,CAAC,IAAI,IAAI,MAAM;AAElG,QAAI,UAAU;AACd,WAAO,UAAU,QAAQ,QAAQ;AAC/B,YAAM,OAAO,QAAQ,UAAU,CAAC;AAChC,UAAI,MAAM;AACR,cAAM,WAAW,KAAK,IAAI,KAAK,aAAa,QAAQ,OAAO,EAAE,UAAU;AACvE,YAAI,WAAW,SAAS;AACtB,cAAG,UAAU;AAAG,sBAAU,UAAU;AAAA;AAC/B;AAAA,QACP;AAAA,MACF;AACA;AAAA,IACF;AAEA,cAAU,QAAQ,MAAM,GAAG,UAAQ,CAAC;AACpC,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAWA,MAAM,uBAAuB,SAAS;AACpC,QAAI,UAAU,CAAC;AACf,UAAM,cAAe,KAAK,OAAO,SAAS,qBAAqB,uBAAwB,KAAK;AAC5F,UAAM,YAAY,cAAc,KAAK,OAAO,SAAS,gBAAgB,IAAI;AACzE,QAAI,aAAa;AACjB,aAAS,IAAI,GAAG,IAAI,QAAQ,QAAQ,KAAK;AACvC,UAAI,QAAQ,UAAU;AACpB;AACF,UAAI,cAAc;AAChB;AACF,UAAI,OAAO,QAAQ,CAAC,EAAE,SAAS;AAC7B;AAEF,YAAM,cAAc,QAAQ,CAAC,EAAE,KAAK,QAAQ,MAAM,KAAK,EAAE,QAAQ,OAAO,EAAE,EAAE,QAAQ,OAAO,KAAK;AAChG,UAAI,cAAc,GAAG;AAAA;AAErB,YAAM,sBAAsB,YAAY,aAAa,YAAY;AACjE,UAAI,QAAQ,CAAC,EAAE,KAAK,QAAQ,GAAG,MAAM,IAAI;AACvC,uBAAe,MAAM,KAAK,OAAO,gBAAgB,QAAQ,CAAC,EAAE,MAAM,EAAE,WAAW,oBAAoB,CAAC;AAAA,MACtG,OAAO;AACL,uBAAe,MAAM,KAAK,OAAO,eAAe,QAAQ,CAAC,EAAE,MAAM,EAAE,WAAW,oBAAoB,CAAC;AAAA,MACrG;AAEA,oBAAc,YAAY;AAE1B,cAAQ,KAAK;AAAA,QACX,MAAM,QAAQ,CAAC,EAAE;AAAA,QACjB,MAAM;AAAA,MACR,CAAC;AAAA,IACH;AAEA,YAAQ,IAAI,sBAAsB,QAAQ,MAAM;AAEhD,YAAQ,IAAI,4BAA4B,KAAK,MAAM,aAAa,GAAG,CAAC;AAEpE,SAAK,KAAK,UAAU,4EAA4E,QAAQ,wIAAwI,kBAAkB,KAAK,OAAO,SAAS,QAAQ,EAAE;AACjS,aAAQ,IAAI,GAAG,IAAI,QAAQ,QAAQ,KAAK;AACtC,WAAK,KAAK,WAAW;AAAA,YAAe,IAAE;AAAA,EAAS,QAAQ,CAAC,EAAE;AAAA,UAAiB,IAAE;AAAA,IAC/E;AACA,WAAO,KAAK,KAAK;AAAA,EACnB;AAGF;AAEA,SAAS,cAAc,QAAM,iBAAiB;AAC5C,QAAM,eAAe;AAAA,IACnB,qBAAqB;AAAA,IACrB,SAAS;AAAA,IACT,iBAAiB;AAAA,IACjB,sBAAsB;AAAA,EACxB;AACA,SAAO,aAAa,KAAK;AAC3B;AAaA,IAAM,4BAAN,MAAgC;AAAA,EAC9B,YAAY,QAAQ;AAClB,SAAK,MAAM,OAAO;AAClB,SAAK,SAAS;AACd,SAAK,UAAU;AACf,SAAK,UAAU,CAAC;AAChB,SAAK,UAAU;AACf,SAAK,MAAM;AACX,SAAK,SAAS,CAAC;AAAA,EACjB;AAAA,EACA,MAAM,YAAY;AAEhB,QAAI,KAAK,OAAO,WAAW;AAAG;AAG9B,QAAI,CAAE,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,0BAA0B,GAAI;AACtE,YAAM,KAAK,IAAI,MAAM,QAAQ,MAAM,0BAA0B;AAAA,IAC/D;AAEA,QAAI,CAAC,KAAK,SAAS;AACjB,WAAK,UAAU,KAAK,KAAK,IAAI,WAAM,KAAK,qBAAqB;AAAA,IAC/D;AAEA,QAAI,CAAC,KAAK,QAAQ,MAAM,qBAAqB,GAAG;AAC9C,cAAQ,IAAI,sBAAsB,KAAK,OAAO;AAC9C,UAAI,SAAS,OAAO,gEAAgE,KAAK,UAAU,GAAG;AAAA,IACxG;AAEA,UAAM,YAAY,KAAK,UAAU;AACjC,SAAK,IAAI,MAAM,QAAQ;AAAA,MACrB,8BAA8B;AAAA,MAC9B,KAAK,UAAU,KAAK,QAAQ,MAAM,CAAC;AAAA,IACrC;AAAA,EACF;AAAA,EACA,MAAM,UAAU,SAAS;AACvB,SAAK,UAAU;AAGf,UAAM,YAAY,KAAK,UAAU;AAEjC,QAAI,YAAY,MAAM,KAAK,IAAI,MAAM,QAAQ;AAAA,MAC3C,8BAA8B;AAAA,IAChC;AAEA,SAAK,SAAS,KAAK,MAAM,SAAS;AAElC,SAAK,UAAU,KAAK,gBAAgB;AAAA,EAKtC;AAAA;AAAA;AAAA,EAGA,gBAAgB,yBAAuB,CAAC,GAAG;AAEzC,QAAG,uBAAuB,WAAW,GAAE;AACrC,WAAK,UAAU,KAAK,OAAO,IAAI,UAAQ;AACrC,eAAO,KAAK,KAAK,SAAS,CAAC;AAAA,MAC7B,CAAC;AAAA,IACH,OAAK;AAGH,UAAI,uBAAuB,CAAC;AAC5B,eAAQ,IAAI,GAAG,IAAI,uBAAuB,QAAQ,KAAI;AACpD,6BAAqB,uBAAuB,CAAC,EAAE,CAAC,CAAC,IAAI,uBAAuB,CAAC,EAAE,CAAC;AAAA,MAClF;AAEA,WAAK,UAAU,KAAK,OAAO,IAAI,CAAC,MAAM,eAAe;AAEnD,YAAG,qBAAqB,UAAU,MAAM,QAAU;AAChD,iBAAO,KAAK,qBAAqB,UAAU,CAAC;AAAA,QAC9C;AAEA,eAAO,KAAK,KAAK,SAAS,CAAC;AAAA,MAC7B,CAAC;AAAA,IACH;AAEA,SAAK,UAAU,KAAK,QAAQ,IAAI,aAAW;AACzC,aAAO;AAAA,QACL,MAAM,QAAQ;AAAA,QACd,SAAS,QAAQ;AAAA,MACnB;AAAA,IACF,CAAC;AACD,WAAO,KAAK;AAAA,EACd;AAAA,EACA,OAAO;AAEL,WAAO,KAAK,OAAO,KAAK,OAAO,SAAS,CAAC,EAAE,KAAK,OAAO,KAAK,OAAO,SAAS,CAAC,EAAE,SAAS,CAAC;AAAA,EAC3F;AAAA,EACA,YAAY;AACV,WAAO,KAAK,KAAK,EAAE;AAAA,EACrB;AAAA;AAAA,EAEA,eAAe;AACb,WAAO,KAAK,KAAK,EAAE;AAAA,EACrB;AAAA;AAAA;AAAA,EAGA,sBAAsB,SAAS,OAAK,IAAI;AAEtC,QAAG,KAAK,SAAQ;AACd,cAAQ,UAAU,KAAK;AACvB,WAAK,UAAU;AAAA,IACjB;AACA,QAAG,KAAK,KAAI;AACV,cAAQ,MAAM,KAAK;AACnB,WAAK,MAAM;AAAA,IACb;AACA,QAAI,SAAS,IAAI;AACf,WAAK,OAAO,KAAK,CAAC,OAAO,CAAC;AAAA,IAC5B,OAAK;AAEH,WAAK,OAAO,IAAI,EAAE,KAAK,OAAO;AAAA,IAChC;AAAA,EACF;AAAA,EACA,gBAAe;AACb,SAAK,UAAU;AACf,SAAK,MAAM;AAAA,EACb;AAAA,EACA,MAAM,YAAY,UAAS;AAEzB,QAAI,KAAK,WAAW,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,8BAA8B,KAAK,UAAU,OAAO,GAAG;AAC7G,iBAAW,KAAK,QAAQ,QAAQ,KAAK,KAAK,GAAG,QAAQ;AAErD,YAAM,KAAK,IAAI,MAAM,QAAQ;AAAA,QAC3B,8BAA8B,KAAK,UAAU;AAAA,QAC7C,8BAA8B,WAAW;AAAA,MAC3C;AAEA,WAAK,UAAU;AAAA,IACjB,OAAK;AACH,WAAK,UAAU,WAAW,WAAM,KAAK,qBAAqB;AAE1D,YAAM,KAAK,UAAU;AAAA,IACvB;AAAA,EAEF;AAAA,EAEA,OAAO;AACL,QAAG,KAAK,SAAQ;AAEd,aAAO,KAAK,QAAQ,QAAQ,WAAU,EAAE;AAAA,IAC1C;AACA,WAAO;AAAA,EACT;AAAA,EAEA,uBAAuB;AACrB,YAAO,oBAAI,KAAK,GAAE,YAAY,EAAE,QAAQ,eAAe,GAAG,EAAE,KAAK;AAAA,EACnE;AAAA;AAAA,EAEA,MAAM,+BAA+B,YAAY,WAAW;AAC1D,QAAI,eAAe;AAEnB,UAAM,QAAQ,KAAK,uBAAuB,UAAU;AAEpD,QAAI,YAAY,cAAc,KAAK,OAAO,SAAS,gBAAgB;AACnE,aAAQ,IAAI,GAAG,IAAI,MAAM,QAAQ,KAAI;AAEnC,YAAM,iBAAkB,MAAM,SAAS,IAAI,IAAK,KAAK,MAAM,aAAa,MAAM,SAAS,EAAE,IAAI;AAE7F,YAAM,eAAe,MAAM,KAAK,kBAAkB,MAAM,CAAC,GAAG,EAAC,YAAY,eAAc,CAAC;AACxF,sBAAgB,oBAAoB,MAAM,CAAC,EAAE;AAAA;AAC7C,sBAAgB;AAChB,sBAAgB;AAAA;AAChB,mBAAa,aAAa;AAC1B,UAAG,aAAa;AAAG;AAAA,IACrB;AACA,SAAK,UAAU;AACf,UAAM,SAAS;AAAA,MACb;AAAA,QACE,MAAM;AAAA,QACN,SAAS;AAAA,MACX;AAAA,MACA;AAAA,QACE,MAAM;AAAA,QACN,SAAS;AAAA,MACX;AAAA,IACF;AACA,cAAU,2BAA2B,EAAC,UAAU,QAAQ,aAAa,EAAC,CAAC;AAAA,EACzE;AAAA;AAAA,EAEA,uBAAuB,YAAY;AACjC,QAAG,WAAW,QAAQ,IAAI,MAAM;AAAI,aAAO;AAC3C,QAAG,WAAW,QAAQ,IAAI,MAAM;AAAI,aAAO;AAC3C,WAAO;AAAA,EACT;AAAA;AAAA,EAEA,0BAA0B,YAAY;AACpC,QAAG,WAAW,QAAQ,GAAG,MAAM;AAAI,aAAO;AAC1C,QAAG,WAAW,QAAQ,GAAG,MAAM,WAAW,YAAY,GAAG;AAAG,aAAO;AACnE,WAAO;AAAA,EACT;AAAA;AAAA,EAEA,sBAAsB,YAAY;AAEhC,UAAM,UAAU,KAAK,OAAO,QAAQ,MAAM;AAC1C,UAAM,UAAU,QAAQ,KAAK,CAAC,GAAG,MAAM,EAAE,SAAS,EAAE,MAAM,EAAE,IAAI,YAAU;AAExE,UAAG,WAAW,QAAQ,MAAM,MAAM,IAAG;AAEnC,qBAAa,WAAW,QAAQ,QAAQ,EAAE;AAC1C,eAAO;AAAA,MACT;AACA,aAAO;AAAA,IACT,CAAC,EAAE,OAAO,YAAU,MAAM;AAC1B,YAAQ,IAAI,OAAO;AAEnB,QAAG;AAAS,aAAO;AACnB,WAAO;AAAA,EACT;AAAA;AAAA,EAIA,uBAAuB,YAAY;AACjC,UAAM,UAAU,WAAW,MAAM,gBAAgB;AACjD,YAAQ,IAAI,OAAO;AAEnB,QAAG;AAAS,aAAO,QAAQ,IAAI,WAAS;AACtC,eAAO,KAAK,IAAI,cAAc,qBAAqB,MAAM,QAAQ,MAAM,EAAE,EAAE,QAAQ,MAAM,EAAE,GAAG,GAAG;AAAA,MACnG,CAAC;AACD,WAAO,CAAC;AAAA,EACV;AAAA;AAAA,EAEA,MAAM,kBAAkB,MAAM,OAAK,CAAC,GAAG;AACrC,WAAO;AAAA,MACL,YAAY;AAAA,MACZ,GAAG;AAAA,IACL;AAEA,QAAG,EAAE,gBAAgB,SAAS;AAAQ,aAAO;AAE7C,QAAI,eAAe,MAAM,KAAK,IAAI,MAAM,WAAW,IAAI;AAEvD,QAAG,aAAa,QAAQ,aAAa,IAAI,IAAG;AAE1C,qBAAe,MAAM,KAAK,wBAAwB,cAAc,KAAK,MAAM,IAAI;AAAA,IACjF;AACA,mBAAe,aAAa,UAAU,GAAG,KAAK,UAAU;AAExD,WAAO;AAAA,EACT;AAAA,EAGA,MAAM,wBAAwB,cAAc,WAAW,OAAK,CAAC,GAAG;AAC9D,WAAO;AAAA,MACL,YAAY;AAAA,MACZ,GAAG;AAAA,IACL;AAEA,UAAM,eAAe,OAAO,aAAa;AAEzC,QAAG,CAAC;AAAc,aAAO;AACzB,UAAM,uBAAuB,aAAa,MAAM,uBAAuB;AAEvE,aAAS,IAAI,GAAG,IAAI,qBAAqB,QAAQ,KAAK;AAEpD,UAAG,KAAK,cAAc,KAAK,aAAa,aAAa,QAAQ,qBAAqB,CAAC,CAAC;AAAG;AAEvF,YAAM,sBAAsB,qBAAqB,CAAC;AAElD,YAAM,8BAA8B,oBAAoB,QAAQ,eAAe,EAAE,EAAE,QAAQ,OAAO,EAAE;AAEpG,YAAM,wBAAwB,MAAM,aAAa,cAAc,6BAA6B,WAAW,IAAI;AAE3G,UAAI,sBAAsB,YAAY;AACpC,uBAAe,aAAa,QAAQ,qBAAqB,sBAAsB,KAAK;AAAA,MACtF;AAAA,IACF;AACA,WAAO;AAAA,EACT;AACF;AAEA,IAAM,mCAAN,cAA+C,SAAS,kBAAkB;AAAA,EACxE,YAAY,KAAK,MAAM,OAAO;AAC5B,UAAM,GAAG;AACT,SAAK,MAAM;AACX,SAAK,OAAO;AACZ,SAAK,eAAe,oCAAoC;AAAA,EAC1D;AAAA,EACA,WAAW;AACT,QAAI,CAAC,KAAK,KAAK,OAAO;AACpB,aAAO,CAAC;AAAA,IACV;AACA,WAAO,KAAK,KAAK;AAAA,EACnB;AAAA,EACA,YAAY,MAAM;AAEhB,QAAG,KAAK,QAAQ,UAAU,MAAM,IAAG;AACjC,WAAK,QAAQ,WAAU,EAAE;AAAA,IAC3B;AACA,WAAO;AAAA,EACT;AAAA,EACA,aAAa,SAAS;AACpB,SAAK,KAAK,UAAU,OAAO;AAAA,EAC7B;AACF;AAGA,IAAM,kCAAN,cAA8C,SAAS,kBAAkB;AAAA,EACvE,YAAY,KAAK,MAAM;AACrB,UAAM,GAAG;AACT,SAAK,MAAM;AACX,SAAK,OAAO;AACZ,SAAK,eAAe,4BAA4B;AAAA,EAClD;AAAA,EACA,WAAW;AAET,WAAO,KAAK,IAAI,MAAM,iBAAiB,EAAE,KAAK,CAAC,GAAG,MAAM,EAAE,SAAS,cAAc,EAAE,QAAQ,CAAC;AAAA,EAC9F;AAAA,EACA,YAAY,MAAM;AAChB,WAAO,KAAK;AAAA,EACd;AAAA,EACA,aAAa,MAAM;AACjB,SAAK,KAAK,iBAAiB,KAAK,WAAW,KAAK;AAAA,EAClD;AACF;AAEA,IAAM,oCAAN,cAAgD,SAAS,kBAAkB;AAAA,EACzE,YAAY,KAAK,MAAM;AACrB,UAAM,GAAG;AACT,SAAK,MAAM;AACX,SAAK,OAAO;AACZ,SAAK,eAAe,8BAA8B;AAAA,EACpD;AAAA,EACA,WAAW;AACT,WAAO,KAAK,KAAK,OAAO;AAAA,EAC1B;AAAA,EACA,YAAY,MAAM;AAChB,WAAO;AAAA,EACT;AAAA,EACA,aAAa,QAAQ;AACnB,SAAK,KAAK,iBAAiB,SAAS,IAAI;AAAA,EAC1C;AACF;AAIA,IAAM,aAAN,MAAiB;AAAA;AAAA,EAEf,YAAY,KAAK,SAAS;AAExB,cAAU,WAAW,CAAC;AACtB,SAAK,MAAM;AACX,SAAK,SAAS,QAAQ,UAAU;AAChC,SAAK,UAAU,QAAQ,WAAW,CAAC;AACnC,SAAK,UAAU,QAAQ,WAAW;AAClC,SAAK,kBAAkB,QAAQ,mBAAmB;AAClD,SAAK,YAAY,CAAC;AAClB,SAAK,aAAa,KAAK;AACvB,SAAK,WAAW;AAChB,SAAK,QAAQ;AACb,SAAK,MAAM;AACX,SAAK,kBAAkB;AACvB,SAAK,eAAe;AACpB,SAAK,aAAa;AAClB,SAAK,OAAO;AACZ,SAAK,SAAS;AAAA,EAChB;AAAA;AAAA,EAEA,iBAAiB,MAAM,UAAU;AAE/B,QAAI,CAAC,KAAK,UAAU,IAAI,GAAG;AACzB,WAAK,UAAU,IAAI,IAAI,CAAC;AAAA,IAC1B;AAEA,QAAG,KAAK,UAAU,IAAI,EAAE,QAAQ,QAAQ,MAAM,IAAI;AAChD,WAAK,UAAU,IAAI,EAAE,KAAK,QAAQ;AAAA,IACpC;AAAA,EACF;AAAA;AAAA,EAEA,oBAAoB,MAAM,UAAU;AAElC,QAAI,CAAC,KAAK,UAAU,IAAI,GAAG;AACzB;AAAA,IACF;AACA,QAAI,WAAW,CAAC;AAEhB,aAAS,IAAI,GAAG,IAAI,KAAK,UAAU,IAAI,EAAE,QAAQ,KAAK;AAEpD,UAAI,KAAK,UAAU,IAAI,EAAE,CAAC,MAAM,UAAU;AACxC,iBAAS,KAAK,KAAK,UAAU,IAAI,EAAE,CAAC,CAAC;AAAA,MACvC;AAAA,IACF;AAEA,QAAI,KAAK,UAAU,IAAI,EAAE,WAAW,GAAG;AACrC,aAAO,KAAK,UAAU,IAAI;AAAA,IAC5B,OAAO;AACL,WAAK,UAAU,IAAI,IAAI;AAAA,IACzB;AAAA,EACF;AAAA;AAAA,EAEA,cAAc,OAAO;AAEnB,QAAI,CAAC,OAAO;AACV,aAAO;AAAA,IACT;AAEA,UAAM,SAAS;AAEf,QAAI,YAAY,OAAO,MAAM;AAE7B,QAAI,KAAK,eAAe,SAAS,GAAG;AAElC,WAAK,SAAS,EAAE,KAAK,MAAM,KAAK;AAEhC,UAAI,MAAM,kBAAkB;AAC1B,eAAO;AAAA,MACT;AAAA,IACF;AAEA,QAAI,KAAK,UAAU,MAAM,IAAI,GAAG;AAC9B,aAAO,KAAK,UAAU,MAAM,IAAI,EAAE,MAAM,SAAS,UAAU;AACzD,iBAAS,KAAK;AACd,eAAO,CAAC,MAAM;AAAA,MAChB,CAAC;AAAA,IACH;AACA,WAAO;AAAA,EACT;AAAA;AAAA,EAEA,eAAe,OAAO;AAEpB,QAAI,QAAQ,IAAI,YAAY,kBAAkB;AAE9C,UAAM,aAAa;AAEnB,SAAK,aAAa;AAElB,SAAK,cAAc,KAAK;AAAA,EAC1B;AAAA;AAAA,EAEA,iBAAiB,GAAG;AAElB,QAAI,QAAQ,IAAI,YAAY,OAAO;AAEnC,UAAM,OAAO,EAAE,cAAc;AAE7B,SAAK,cAAc,KAAK;AACxB,SAAK,MAAM;AAAA,EACb;AAAA;AAAA,EAEA,eAAe,GAAG;AAEhB,QAAI,QAAQ,IAAI,YAAY,OAAO;AAEnC,SAAK,MAAM;AAAA,EACb;AAAA;AAAA,EAEA,kBAAkB,GAAG;AAEnB,QAAI,CAAC,KAAK,KAAK;AACb;AAAA,IACF;AAEA,QAAI,KAAK,IAAI,WAAW,KAAK;AAE3B,WAAK,iBAAiB,CAAC;AACvB;AAAA,IACF;AAEA,QAAI,KAAK,eAAe,KAAK,YAAY;AAEvC,WAAK,cAAc,IAAI,YAAY,MAAM,CAAC;AAE1C,WAAK,eAAe,KAAK,IAAI;AAAA,IAC/B;AAEA,QAAI,OAAO,KAAK,IAAI,aAAa,UAAU,KAAK,QAAQ;AAExD,SAAK,YAAY,KAAK;AAEtB,SAAK,MAAM,kBAAkB,EAAE,QAAQ,SAAS,MAAK;AACnD,UAAG,KAAK,KAAK,EAAE,WAAW,GAAG;AAC3B,aAAK,cAAc,KAAK,iBAAiB,KAAK,MAAM,KAAK,CAAC,CAAC;AAC3D,aAAK,QAAQ;AAAA,MACf,OAAO;AACL,aAAK,SAAS;AAAA,MAChB;AAAA,IACF,EAAE,KAAK,IAAI,CAAC;AAAA,EACd;AAAA;AAAA,EAEA,gBAAgB,GAAG;AACjB,SAAK,kBAAkB,CAAC;AAExB,SAAK,cAAc,KAAK,iBAAiB,KAAK,KAAK,CAAC;AACpD,SAAK,QAAQ;AAAA,EACf;AAAA;AAAA,EAEA,iBAAiB,OAAO;AAEtB,QAAI,CAAC,SAAS,MAAM,WAAW,GAAG;AAChC,aAAO;AAAA,IACT;AAEA,QAAI,IAAI,EAAC,IAAI,MAAM,OAAO,MAAM,MAAM,IAAI,OAAO,UAAS;AAE1D,UAAM,MAAM,cAAc,EAAE,QAAQ,SAAS,MAAM;AACjD,aAAO,KAAK,UAAU;AACtB,UAAI,QAAQ,KAAK,QAAQ,KAAK,eAAe;AAC7C,UAAG,SAAS,GAAG;AACb;AAAA,MACF;AAEA,UAAI,QAAQ,KAAK,UAAU,GAAG,KAAK;AACnC,UAAG,EAAE,SAAS,IAAI;AAChB;AAAA,MACF;AAEA,UAAI,QAAQ,KAAK,UAAU,QAAQ,CAAC,EAAE,SAAS;AAC/C,UAAG,UAAU,QAAQ;AACnB,UAAE,KAAK,KAAK;AAAA,MACd,OAAO;AACL,UAAE,KAAK,IAAI;AAAA,MACb;AAAA,IACF,EAAE,KAAK,IAAI,CAAC;AAEZ,QAAI,QAAQ,IAAI,YAAY,EAAE,KAAK;AACnC,UAAM,OAAO,EAAE;AACf,UAAM,KAAK,EAAE;AACb,WAAO;AAAA,EACT;AAAA;AAAA,EAEA,qBAAqB;AACnB,QAAG,CAAC,KAAK,KAAK;AACZ;AAAA,IACF;AACA,QAAG,KAAK,IAAI,eAAe,eAAe,MAAM;AAC9C,WAAK,eAAe,KAAK,MAAM;AAAA,IACjC;AAAA,EACF;AAAA;AAAA,EAEA,SAAS;AAEP,SAAK,eAAe,KAAK,UAAU;AAEnC,SAAK,MAAM,IAAI,eAAe;AAE9B,SAAK,IAAI,iBAAiB,YAAY,KAAK,kBAAkB,KAAK,IAAI,CAAC;AAEvE,SAAK,IAAI,iBAAiB,QAAQ,KAAK,gBAAgB,KAAK,IAAI,CAAC;AAEjE,SAAK,IAAI,iBAAiB,oBAAoB,KAAK,mBAAmB,KAAK,IAAI,CAAC;AAEhF,SAAK,IAAI,iBAAiB,SAAS,KAAK,iBAAiB,KAAK,IAAI,CAAC;AAEnE,SAAK,IAAI,iBAAiB,SAAS,KAAK,eAAe,KAAK,IAAI,CAAC;AAEjE,SAAK,IAAI,KAAK,KAAK,QAAQ,KAAK,GAAG;AAEnC,aAAS,UAAU,KAAK,SAAS;AAC/B,WAAK,IAAI,iBAAiB,QAAQ,KAAK,QAAQ,MAAM,CAAC;AAAA,IACxD;AAEA,SAAK,IAAI,kBAAkB,KAAK;AAEhC,SAAK,IAAI,KAAK,KAAK,OAAO;AAAA,EAC5B;AAAA;AAAA,EAEA,QAAQ;AACN,QAAG,KAAK,eAAe,KAAK,QAAQ;AAClC;AAAA,IACF;AACA,SAAK,IAAI,MAAM;AACf,SAAK,MAAM;AACX,SAAK,eAAe,KAAK,MAAM;AAAA,EACjC;AACF;AAEA,OAAO,UAAU;",
  "names": ["exports", "module", "VecLite", "line_limit", "item", "link", "file_link", "file_link_list"]
}
 diff --git a/main.js b/main.js new file mode 100644 index 00000000..ecff7aeb --- /dev/null +++ b/main.js @@ -0,0 +1,3382 @@ +// src/index.js +var Obsidian = require("obsidian"); +var VecLite = class { + constructor(config) { + this.config = { + file_name: "embeddings-3.json", + folder_path: ".vec_lite", + exists_adapter: null, + mkdir_adapter: null, + read_adapter: null, + rename_adapter: null, + stat_adapter: null, + write_adapter: null, + ...config + }; + this.file_name = this.config.file_name; + this.folder_path = config.folder_path; + this.file_path = this.folder_path + "/" + this.file_name; + this.embeddings = false; + } + async file_exists(path) { + if (this.config.exists_adapter) { + return await this.config.exists_adapter(path); + } else { + throw new Error("exists_adapter not set"); + } + } + async mkdir(path) { + if (this.config.mkdir_adapter) { + return await this.config.mkdir_adapter(path); + } else { + throw new Error("mkdir_adapter not set"); + } + } + async read_file(path) { + if (this.config.read_adapter) { + return await this.config.read_adapter(path); + } else { + throw new Error("read_adapter not set"); + } + } + async rename(old_path, new_path) { + if (this.config.rename_adapter) { + return await this.config.rename_adapter(old_path, new_path); + } else { + throw new Error("rename_adapter not set"); + } + } + async stat(path) { + if (this.config.stat_adapter) { + return await this.config.stat_adapter(path); + } else { + throw new Error("stat_adapter not set"); + } + } + async write_file(path, data) { + if (this.config.write_adapter) { + return await this.config.write_adapter(path, data); + } else { + throw new Error("write_adapter not set"); + } + } + async load(retries = 0) { + try { + const embeddings_file = await this.read_file(this.file_path); + this.embeddings = JSON.parse(embeddings_file); + console.log("loaded embeddings file: " + this.file_path); + return true; + } catch (error) { + if (retries < 3) { + console.log("retrying load()"); + await new Promise((r) => setTimeout(r, 1e3 + 1e3 * retries)); + return await this.load(retries + 1); + } + console.log( + "failed to load embeddings file, prompt user to initiate bulk embed" + ); + return false; + } + } + async init_embeddings_file() { + if (!await this.file_exists(this.folder_path)) { + await this.mkdir(this.folder_path); + console.log("created folder: " + this.folder_path); + } else { + console.log("folder already exists: " + this.folder_path); + } + if (!await this.file_exists(this.file_path)) { + await this.write_file(this.file_path, "{}"); + console.log("created embeddings file: " + this.file_path); + } else { + console.log("embeddings file already exists: " + this.file_path); + } + } + async save() { + const embeddings = JSON.stringify(this.embeddings); + const embeddings_file_exists = await this.file_exists(this.file_path); + if (embeddings_file_exists) { + const new_file_size = embeddings.length; + const existing_file_size = await this.stat(this.file_path).then( + (stat) => stat.size + ); + if (new_file_size > existing_file_size * 0.5) { + await this.write_file(this.file_path, embeddings); + console.log("embeddings file size: " + new_file_size + " bytes"); + } else { + const warning_message = [ + "Warning: New embeddings file size is significantly smaller than existing embeddings file size.", + "Aborting to prevent possible loss of embeddings data.", + "New file size: " + new_file_size + " bytes.", + "Existing file size: " + existing_file_size + " bytes.", + "Restarting Obsidian may fix this." + ]; + console.log(warning_message.join(" ")); + await this.write_file( + this.folder_path + "/unsaved-embeddings.json", + embeddings + ); + throw new Error( + "Error: New embeddings file size is significantly smaller than existing embeddings file size. Aborting to prevent possible loss of embeddings data." + ); + } + } else { + await this.init_embeddings_file(); + return await this.save(); + } + return true; + } + cos_sim(vector1, vector2) { + let dotProduct = 0; + let normA = 0; + let normB = 0; + for (let i = 0; i < vector1.length; i++) { + dotProduct += vector1[i] * vector2[i]; + normA += vector1[i] * vector1[i]; + normB += vector2[i] * vector2[i]; + } + if (normA === 0 || normB === 0) { + return 0; + } else { + return dotProduct / (Math.sqrt(normA) * Math.sqrt(normB)); + } + } + nearest(to_vec, filter = {}) { + filter = { + results_count: 30, + ...filter + }; + let nearest = []; + const from_keys = Object.keys(this.embeddings); + for (let i = 0; i < from_keys.length; i++) { + if (filter.skip_sections) { + const from_path = this.embeddings[from_keys[i]].meta.path; + if (from_path.indexOf("#") > -1) + continue; + } + if (filter.skip_key) { + if (filter.skip_key === from_keys[i]) + continue; + if (filter.skip_key === this.embeddings[from_keys[i]].meta.parent) + continue; + } + if (filter.path_begins_with) { + if (typeof filter.path_begins_with === "string" && !this.embeddings[from_keys[i]].meta.path.startsWith( + filter.path_begins_with + )) + continue; + if (Array.isArray(filter.path_begins_with) && !filter.path_begins_with.some( + (path) => this.embeddings[from_keys[i]].meta.path.startsWith(path) + )) + continue; + } + nearest.push({ + link: this.embeddings[from_keys[i]].meta.path, + similarity: this.cos_sim(to_vec, this.embeddings[from_keys[i]].vec), + size: this.embeddings[from_keys[i]].meta.size + }); + } + nearest.sort(function(a, b) { + return b.similarity - a.similarity; + }); + nearest = nearest.slice(0, filter.results_count); + return nearest; + } + find_nearest_embeddings(to_vec, filter = {}) { + const default_filter = { + max: this.max_sources + }; + filter = { ...default_filter, ...filter }; + if (Array.isArray(to_vec) && to_vec.length !== this.vec_len) { + this.nearest = {}; + for (let i = 0; i < to_vec.length; i++) { + this.find_nearest_embeddings(to_vec[i], { + max: Math.floor(filter.max / to_vec.length) + }); + } + } else { + const from_keys = Object.keys(this.embeddings); + for (let i = 0; i < from_keys.length; i++) { + if (this.validate_type(this.embeddings[from_keys[i]])) + continue; + const sim = this.computeCosineSimilarity( + to_vec, + this.embeddings[from_keys[i]].vec + ); + if (this.nearest[from_keys[i]]) { + this.nearest[from_keys[i]] += sim; + } else { + this.nearest[from_keys[i]] = sim; + } + } + } + let nearest = Object.keys(this.nearest).map((key) => { + return { + key, + similarity: this.nearest[key] + }; + }); + nearest = this.sort_by_similarity(nearest); + nearest = nearest.slice(0, filter.max); + nearest = nearest.map((item) => { + return { + link: this.embeddings[item.key].meta.path, + similarity: item.similarity, + len: this.embeddings[item.key].meta.len || this.embeddings[item.key].meta.size + }; + }); + return nearest; + } + sort_by_similarity(nearest) { + return nearest.sort(function(a, b) { + const a_score = a.similarity; + const b_score = b.similarity; + if (a_score > b_score) + return -1; + if (a_score < b_score) + return 1; + return 0; + }); + } + // check if key from embeddings exists in files + clean_up_embeddings(files) { + console.log("cleaning up embeddings"); + const keys = Object.keys(this.embeddings); + let deleted_embeddings = 0; + for (const key of keys) { + const path = this.embeddings[key].meta.path; + if (!files.find((file) => path.startsWith(file.path))) { + delete this.embeddings[key]; + deleted_embeddings++; + continue; + } + if (path.indexOf("#") > -1) { + const parent_key = this.embeddings[key].meta.parent; + if (!this.embeddings[parent_key]) { + delete this.embeddings[key]; + deleted_embeddings++; + continue; + } + if (!this.embeddings[parent_key].meta) { + delete this.embeddings[key]; + deleted_embeddings++; + continue; + } + if (this.embeddings[parent_key].meta.children && this.embeddings[parent_key].meta.children.indexOf(key) < 0) { + delete this.embeddings[key]; + deleted_embeddings++; + continue; + } + } + } + return { deleted_embeddings, total_embeddings: keys.length }; + } + get(key) { + return this.embeddings[key] || null; + } + get_meta(key) { + const embedding = this.get(key); + if (embedding && embedding.meta) { + return embedding.meta; + } + return null; + } + get_mtime(key) { + const meta = this.get_meta(key); + if (meta && meta.mtime) { + return meta.mtime; + } + return null; + } + get_hash(key) { + const meta = this.get_meta(key); + if (meta && meta.hash) { + return meta.hash; + } + return null; + } + get_size(key) { + const meta = this.get_meta(key); + if (meta && meta.size) { + return meta.size; + } + return null; + } + get_children(key) { + const meta = this.get_meta(key); + if (meta && meta.children) { + return meta.children; + } + return null; + } + get_vec(key) { + const embedding = this.get(key); + if (embedding && embedding.vec) { + return embedding.vec; + } + return null; + } + save_embedding(key, vec, meta) { + this.embeddings[key] = { + vec, + meta + }; + } + mtime_is_current(key, source_mtime) { + const mtime = this.get_mtime(key); + if (mtime && mtime >= source_mtime) { + return true; + } + return false; + } + async force_refresh() { + this.embeddings = null; + this.embeddings = {}; + let current_datetime = Math.floor(Date.now() / 1e3); + await this.rename( + this.file_path, + this.folder_path + "/embeddings-" + current_datetime + ".json" + ); + await this.init_embeddings_file(); + } +}; +var DEFAULT_SETTINGS = { + api_key: "", + chat_open: true, + file_exclusions: "", + folder_exclusions: "", + header_exclusions: "", + path_only: "", + show_full_path: false, + cut_off_frontmatter: false, + expanded_view: true, + group_nearest_by_file: false, + language: "en", + log_render: false, + log_render_files: false, + recently_sent_retry_notice: false, + skip_sections: false, + smart_chat_model: "gpt-3.5-turbo-16k", + view_open: true, + version: "", + open_in_big_view: false +}; +var MAX_EMBED_STRING_LENGTH = 25e3; +var VERSION; +var SUPPORTED_FILE_TYPES = ["md", "canvas"]; +var SMART_TRANSLATION = { + "en": { + "pronous": ["my", "I", "me", "mine", "our", "ours", "us", "we"], + "prompt": "Based on your notes", + "initial_message": "Hi, I'm ChatGPT with access to your notes via Smart Connections. Ask me a question about your notes and I'll try to answer it.", + "try_placeholder": `Try "Based on my notes" or "Summarize [[this note]]" or "Important tasks in /folder/"` + }, + "es": { + "pronous": ["mi", "yo", "m\xED", "t\xFA"], + "prompt": "Bas\xE1ndose en sus notas", + "initial_message": "Hola, soy ChatGPT con acceso a tus apuntes a trav\xE9s de Smart Connections. Hazme una pregunta sobre tus apuntes e intentar\xE9 responderte.", + "try_placeholder": `Prueba "Basado en mis notas" o "Resumen [[esta nota]]" o "Tareas importantes en /carpeta/"` + }, + "fr": { + "pronous": ["me", "mon", "ma", "mes", "moi", "nous", "notre", "nos", "je", "j'", "m'"], + "prompt": "D'apr\xE8s vos notes", + "initial_message": "Bonjour, je suis ChatGPT et j'ai acc\xE8s \xE0 vos notes via Smart Connections. Posez-moi une question sur vos notes et j'essaierai d'y r\xE9pondre.", + "try_placeholder": `Essayez "D'apr\xE8s mes notes" ou "R\xE9sume [[cette note]]" ou "T\xE2ches importantes dans /dossier/"` + }, + "de": { + "pronous": ["mein", "meine", "meinen", "meiner", "meines", "mir", "uns", "unser", "unseren", "unserer", "unseres"], + "prompt": "Basierend auf Ihren Notizen", + "initial_message": "Hallo, ich bin ChatGPT und habe \xFCber Smart Connections Zugang zu Ihren Notizen. Stellen Sie mir eine Frage zu Ihren Notizen und ich werde versuchen, sie zu beantworten.", + "try_placeholder": `Versuchen Sie "Basierend auf meinen Notizen" oder "Zusammenfassen [[dieser Notiz]]" oder "Wichtige Aufgaben im /Ordner/"` + }, + "it": { + "pronous": ["mio", "mia", "miei", "mie", "noi", "nostro", "nostri", "nostra", "nostre"], + "prompt": "Sulla base degli appunti", + "initial_message": "Ciao, sono ChatGPT e ho accesso ai tuoi appunti tramite Smart Connections. Fatemi una domanda sui vostri appunti e cercher\xF2 di rispondervi.", + "try_placeholder": `Prova "Sulla base dei miei appunti" o "Riassumi [[questo appunto]]" o "Compiti importanti in /cartella/"` + } +}; +var crypto = require("crypto"); +function md5(str) { + return crypto.createHash("md5").update(str).digest("hex"); +} +var SmartConnectionsPlugin = class extends Obsidian.Plugin { + // constructor + constructor() { + super(...arguments); + this.api = null; + this.embeddings_loaded = false; + this.file_exclusions = []; + this.folders = []; + this.has_new_embeddings = false; + this.header_exclusions = []; + this.nearest_cache = {}; + this.path_only = []; + this.render_log = {}; + this.render_log.deleted_embeddings = 0; + this.render_log.exclusions_logs = {}; + this.render_log.failed_embeddings = []; + this.render_log.files = []; + this.render_log.new_embeddings = 0; + this.render_log.skipped_low_delta = {}; + this.render_log.token_usage = 0; + this.render_log.tokens_saved_by_cache = 0; + this.retry_notice_timeout = null; + this.save_timeout = null; + this.sc_branding = {}; + this.self_ref_kw_regex = null; + this.update_available = false; + } + async onload() { + this.app.workspace.onLayoutReady(this.initialize.bind(this)); + } + onunload() { + this.output_render_log(); + console.log("unloading plugin"); + this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE); + this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE); + } + async initialize() { + console.log("Loading Smart Connections plugin"); + VERSION = this.manifest.version; + await this.loadSettings(); + setTimeout(this.check_for_update.bind(this), 3e3); + setInterval(this.check_for_update.bind(this), 108e5); + this.addIcon(); + this.addCommand({ + id: "sc-find-notes", + name: "Find: Make Smart Connections", + icon: "pencil_icon", + hotkeys: [], + // editorCallback: async (editor) => { + editorCallback: async (editor) => { + if (editor.somethingSelected()) { + let selected_text = editor.getSelection(); + await this.make_connections(selected_text); + } else { + this.nearest_cache = {}; + await this.make_connections(); + } + } + }); + this.addCommand({ + id: "smart-connections-view", + name: "Open: View Smart Connections", + callback: () => { + this.open_view(); + } + }); + this.addCommand({ + id: "smart-connections-chat", + name: "Open: Smart Chat Conversation", + callback: () => { + this.open_chat(); + } + }); + this.addCommand({ + id: "smart-connections-random", + name: "Open: Random Note from Smart Connections", + callback: () => { + this.open_random_note(); + } + }); + this.addSettingTab(new SmartConnectionsSettingsTab(this.app, this)); + this.registerView(SMART_CONNECTIONS_VIEW_TYPE, (leaf) => new SmartConnectionsView(leaf, this)); + this.registerView(SMART_CONNECTIONS_CHAT_VIEW_TYPE, (leaf) => new SmartConnectionsChatView(leaf, this)); + this.registerMarkdownCodeBlockProcessor("smart-connections", this.render_code_block.bind(this)); + if (this.settings.view_open) { + this.open_view(); + } + if (this.settings.chat_open) { + this.open_chat(); + } + if (this.settings.version !== VERSION) { + this.settings.version = VERSION; + await this.saveSettings(); + this.open_view(); + } + this.add_to_gitignore(); + this.api = new ScSearchApi(this.app, this); + (window["SmartSearchApi"] = this.api) && this.register(() => delete window["SmartSearchApi"]); + } + async init_vecs() { + this.smart_vec_lite = new VecLite({ + folder_path: ".smart-connections", + exists_adapter: this.app.vault.adapter.exists.bind(this.app.vault.adapter), + mkdir_adapter: this.app.vault.adapter.mkdir.bind(this.app.vault.adapter), + read_adapter: this.app.vault.adapter.read.bind(this.app.vault.adapter), + rename_adapter: this.app.vault.adapter.rename.bind(this.app.vault.adapter), + stat_adapter: this.app.vault.adapter.stat.bind(this.app.vault.adapter), + write_adapter: this.app.vault.adapter.write.bind(this.app.vault.adapter) + }); + this.embeddings_loaded = await this.smart_vec_lite.load(); + return this.embeddings_loaded; + } + async update_to_v2() { + if (!this.settings.license_key) + return new Obsidian.Notice("[Smart Connections] Supporter license key required for early access to V2"); + const v2 = await (0, Obsidian.requestUrl)({ + url: "https://sync.smartconnections.app/download_v2", + method: "POST", + headers: { + "Content-Type": "application/json" + }, + body: JSON.stringify({ + license_key: this.settings.license_key + }) + }); + if (v2.status !== 200) + return console.error("Error downloading version 2", v2); + console.log(v2); + await this.app.vault.adapter.write(".obsidian/plugins/smart-connections/main.js", v2.json.main); + await this.app.vault.adapter.write(".obsidian/plugins/smart-connections/manifest.json", v2.json.manifest); + await this.app.vault.adapter.write(".obsidian/plugins/smart-connections/styles.css", v2.json.styles); + window.restart_plugin = async (id) => { + console.log("restarting plugin", id); + await window.app.plugins.disablePlugin(id); + await window.app.plugins.enablePlugin(id); + console.log("plugin restarted", id); + }; + window.restart_plugin(this.manifest.id); + } + async loadSettings() { + this.settings = Object.assign({}, DEFAULT_SETTINGS, await this.loadData()); + if (this.settings.file_exclusions && this.settings.file_exclusions.length > 0) { + this.file_exclusions = this.settings.file_exclusions.split(/[,\n\s]/).filter((file) => file.length > 0).map((file) => { + return file.trim(); + }); + } + if (this.settings.folder_exclusions && this.settings.folder_exclusions.length > 0) { + const folder_exclusions = this.settings.folder_exclusions.split(/[,\n\s]/).filter((file) => file.length > 0).map((folder) => { + folder = folder.trim(); + return folder.slice(-1) === "/" ? folder : `${folder}/`; + }); + this.file_exclusions = this.file_exclusions.concat(folder_exclusions); + } + if (this.settings.header_exclusions && this.settings.header_exclusions.length > 0) { + this.header_exclusions = this.settings.header_exclusions.split(/[\s\n,]/).filter((file) => file.length > 0).map((header) => { + return header.trim(); + }); + } + if (this.settings.path_only && this.settings.path_only.length > 0) { + this.path_only = this.settings.path_only.split(/[\s\n,]/).filter((file) => file.length > 0).map((path) => { + return path.trim(); + }); + } + this.self_ref_kw_regex = new RegExp(`\\b(${SMART_TRANSLATION[this.settings.language].pronous.join("|")})\\b`, "gi"); + await this.load_failed_files(); + } + async saveSettings(rerender = false) { + await this.saveData(this.settings); + await this.loadSettings(); + if (rerender) { + this.nearest_cache = {}; + await this.make_connections(); + } + } + // check for update + async check_for_update() { + try { + const response = await (0, Obsidian.requestUrl)({ + url: "https://api.github.com/repos/brianpetro/obsidian-smart-connections/releases/latest", + method: "GET", + headers: { + "Content-Type": "application/json" + }, + contentType: "application/json" + }); + const latest_release = JSON.parse(response.text).tag_name; + if (latest_release !== VERSION) { + new Obsidian.Notice(`[Smart Connections] A new version is available! (v${latest_release})`); + this.update_available = true; + this.render_brand("all"); + } + } catch (error) { + console.log(error); + } + } + async render_code_block(contents, container, ctx) { + let nearest; + if (contents.trim().length > 0) { + nearest = await this.api.search(contents); + } else { + console.log(ctx); + const file = this.app.vault.getAbstractFileByPath(ctx.sourcePath); + nearest = await this.find_note_connections(file); + } + if (nearest.length) { + this.update_results(container, nearest); + } + } + async make_connections(selected_text = null) { + let view = this.get_view(); + if (!view) { + await this.open_view(); + view = this.get_view(); + } + await view.render_connections(selected_text); + } + addIcon() { + Obsidian.addIcon("smart-connections", ` + + + + + + `); + } + // open random note + async open_random_note() { + const curr_file = this.app.workspace.getActiveFile(); + const curr_key = md5(curr_file.path); + if (typeof this.nearest_cache[curr_key] === "undefined") { + new Obsidian.Notice("[Smart Connections] No Smart Connections found. Open a note to get Smart Connections."); + return; + } + const rand = Math.floor(Math.random() * this.nearest_cache[curr_key].length / 2); + const random_file = this.nearest_cache[curr_key][rand]; + this.open_note(random_file); + } + async open_view() { + if (this.get_view()) { + console.log("Smart Connections view already open"); + return; + } + this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE); + await this.app.workspace.getRightLeaf(false).setViewState({ + type: SMART_CONNECTIONS_VIEW_TYPE, + active: true + }); + this.app.workspace.revealLeaf( + this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE)[0] + ); + } + // source: https://github.com/obsidianmd/obsidian-releases/blob/master/plugin-review.md#avoid-managing-references-to-custom-views + get_view() { + for (let leaf of this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE)) { + if (leaf.view instanceof SmartConnectionsView) { + return leaf.view; + } + } + } + // open chat view + async open_chat(retries = 0) { + if (!this.embeddings_loaded) { + console.log("embeddings not loaded yet"); + if (retries < 3) { + setTimeout(() => { + this.open_chat(retries + 1); + }, 1e3 * (retries + 1)); + return; + } + console.log("embeddings still not loaded, opening smart view"); + this.open_view(); + return; + } + this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE); + if (!this.settings.open_in_big_view) { + await this.app.workspace.getRightLeaf(false).setViewState({ + type: SMART_CONNECTIONS_CHAT_VIEW_TYPE, + active: true + }); + } else { + await this.app.workspace.getLeaf(true).setViewState({ + type: SMART_CONNECTIONS_CHAT_VIEW_TYPE, + active: true + }); + } + this.app.workspace.revealLeaf( + this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE)[0] + ); + } + // get embeddings for all files + async get_all_embeddings() { + const files = (await this.app.vault.getFiles()).filter((file) => file instanceof Obsidian.TFile && (file.extension === "md" || file.extension === "canvas")); + const open_files = this.app.workspace.getLeavesOfType("markdown").map((leaf) => leaf.view.file); + const clean_up_log = this.smart_vec_lite.clean_up_embeddings(files); + if (this.settings.log_render) { + this.render_log.total_files = files.length; + this.render_log.deleted_embeddings = clean_up_log.deleted_embeddings; + this.render_log.total_embeddings = clean_up_log.total_embeddings; + } + let batch_promises = []; + for (let i = 0; i < files.length; i++) { + if (files[i].path.indexOf("#") > -1) { + this.log_exclusion("path contains #"); + continue; + } + if (this.smart_vec_lite.mtime_is_current(md5(files[i].path), files[i].stat.mtime)) { + continue; + } + if (this.settings.failed_files.indexOf(files[i].path) > -1) { + if (this.retry_notice_timeout) { + clearTimeout(this.retry_notice_timeout); + this.retry_notice_timeout = null; + } + if (!this.recently_sent_retry_notice) { + new Obsidian.Notice("Smart Connections: Skipping previously failed file, use button in settings to retry"); + this.recently_sent_retry_notice = true; + setTimeout(() => { + this.recently_sent_retry_notice = false; + }, 6e5); + } + continue; + } + let skip = false; + for (const fileExclusion of this.file_exclusions) { + if (files[i].path.includes(fileExclusion)) { + skip = true; + this.log_exclusion(fileExclusion); + break; + } + } + if (skip) { + continue; + } + if (open_files.indexOf(files[i]) > -1) { + continue; + } + try { + batch_promises.push(this.get_file_embeddings(files[i], false)); + } catch (error) { + console.log(error); + } + if (batch_promises.length > 3) { + await Promise.all(batch_promises); + batch_promises = []; + } + if (i > 0 && i % 100 === 0) { + await this.save_embeddings_to_file(); + } + } + await Promise.all(batch_promises); + await this.save_embeddings_to_file(); + if (this.render_log.failed_embeddings.length > 0) { + await this.save_failed_embeddings(); + } + } + async save_embeddings_to_file(force = false) { + if (!this.has_new_embeddings) { + return; + } + if (!force) { + if (this.save_timeout) { + clearTimeout(this.save_timeout); + this.save_timeout = null; + } + this.save_timeout = setTimeout(() => { + this.save_embeddings_to_file(true); + if (this.save_timeout) { + clearTimeout(this.save_timeout); + this.save_timeout = null; + } + }, 3e4); + console.log("scheduled save"); + return; + } + try { + await this.smart_vec_lite.save(); + this.has_new_embeddings = false; + } catch (error) { + console.log(error); + new Obsidian.Notice("Smart Connections: " + error.message); + } + } + // save failed embeddings to file from render_log.failed_embeddings + async save_failed_embeddings() { + let failed_embeddings = []; + const failed_embeddings_file_exists = await this.app.vault.adapter.exists(".smart-connections/failed-embeddings.txt"); + if (failed_embeddings_file_exists) { + failed_embeddings = await this.app.vault.adapter.read(".smart-connections/failed-embeddings.txt"); + failed_embeddings = failed_embeddings.split("\r\n"); + } + failed_embeddings = failed_embeddings.concat(this.render_log.failed_embeddings); + failed_embeddings = [...new Set(failed_embeddings)]; + failed_embeddings.sort(); + failed_embeddings = failed_embeddings.join("\r\n"); + await this.app.vault.adapter.write(".smart-connections/failed-embeddings.txt", failed_embeddings); + await this.load_failed_files(); + } + // load failed files from failed-embeddings.txt + async load_failed_files() { + const failed_embeddings_file_exists = await this.app.vault.adapter.exists(".smart-connections/failed-embeddings.txt"); + if (!failed_embeddings_file_exists) { + this.settings.failed_files = []; + console.log("No failed files."); + return; + } + const failed_embeddings = await this.app.vault.adapter.read(".smart-connections/failed-embeddings.txt"); + const failed_embeddings_array = failed_embeddings.split("\r\n"); + const failed_files = failed_embeddings_array.map((embedding) => embedding.split("#")[0]).reduce((unique, item) => unique.includes(item) ? unique : [...unique, item], []); + this.settings.failed_files = failed_files; + } + // retry failed embeddings + async retry_failed_files() { + this.settings.failed_files = []; + const failed_embeddings_file_exists = await this.app.vault.adapter.exists(".smart-connections/failed-embeddings.txt"); + if (failed_embeddings_file_exists) { + await this.app.vault.adapter.remove(".smart-connections/failed-embeddings.txt"); + } + await this.get_all_embeddings(); + } + // add .smart-connections to .gitignore to prevent issues with large, frequently updated embeddings file(s) + async add_to_gitignore() { + if (!await this.app.vault.adapter.exists(".gitignore")) { + return; + } + let gitignore_file = await this.app.vault.adapter.read(".gitignore"); + if (gitignore_file.indexOf(".smart-connections") < 0) { + let add_to_gitignore = "\n\n# Ignore Smart Connections folder because embeddings file is large and updated frequently"; + add_to_gitignore += "\n.smart-connections"; + await this.app.vault.adapter.write(".gitignore", gitignore_file + add_to_gitignore); + console.log("added .smart-connections to .gitignore"); + } + } + // force refresh embeddings file but first rename existing embeddings file to .smart-connections/embeddings-YYYY-MM-DD.json + async force_refresh_embeddings_file() { + new Obsidian.Notice("Smart Connections: embeddings file Force Refreshed, making new connections..."); + await this.smart_vec_lite.force_refresh(); + await this.get_all_embeddings(); + this.output_render_log(); + new Obsidian.Notice("Smart Connections: embeddings file Force Refreshed, new connections made."); + } + // get embeddings for embed_input + async get_file_embeddings(curr_file, save = true) { + let req_batch = []; + let blocks = []; + const curr_file_key = md5(curr_file.path); + let file_embed_input = curr_file.path.replace(".md", ""); + file_embed_input = file_embed_input.replace(/\//g, " > "); + let path_only = false; + for (let j = 0; j < this.path_only.length; j++) { + if (curr_file.path.indexOf(this.path_only[j]) > -1) { + path_only = true; + console.log("title only file with matcher: " + this.path_only[j]); + break; + } + } + if (path_only) { + req_batch.push([curr_file_key, file_embed_input, { + mtime: curr_file.stat.mtime, + path: curr_file.path + }]); + await this.get_embeddings_batch(req_batch); + return; + } + if (curr_file.extension === "canvas") { + const canvas_contents = await this.app.vault.cachedRead(curr_file); + if (typeof canvas_contents === "string" && canvas_contents.indexOf("nodes") > -1) { + const canvas_json = JSON.parse(canvas_contents); + for (let j = 0; j < canvas_json.nodes.length; j++) { + if (canvas_json.nodes[j].text) { + file_embed_input += "\n" + canvas_json.nodes[j].text; + } + if (canvas_json.nodes[j].file) { + file_embed_input += "\nLink: " + canvas_json.nodes[j].file; + } + } + } + req_batch.push([curr_file_key, file_embed_input, { + mtime: curr_file.stat.mtime, + path: curr_file.path + }]); + await this.get_embeddings_batch(req_batch); + return; + } + const note_contents = await this.app.vault.cachedRead(curr_file); + let processed_since_last_save = 0; + const note_sections = this.block_parser(note_contents, curr_file.path); + if (note_sections.length > 1) { + for (let j = 0; j < note_sections.length; j++) { + const block_embed_input = note_sections[j].text; + const block_key = md5(note_sections[j].path); + blocks.push(block_key); + if (this.smart_vec_lite.get_size(block_key) === block_embed_input.length) { + continue; + } + if (this.smart_vec_lite.mtime_is_current(block_key, curr_file.stat.mtime)) { + continue; + } + const block_hash = md5(block_embed_input.trim()); + if (this.smart_vec_lite.get_hash(block_key) === block_hash) { + continue; + } + req_batch.push([block_key, block_embed_input, { + // oldmtime: curr_file.stat.mtime, + // get current datetime as unix timestamp + mtime: Date.now(), + hash: block_hash, + parent: curr_file_key, + path: note_sections[j].path, + size: block_embed_input.length + }]); + if (req_batch.length > 9) { + await this.get_embeddings_batch(req_batch); + processed_since_last_save += req_batch.length; + if (processed_since_last_save >= 30) { + await this.save_embeddings_to_file(); + processed_since_last_save = 0; + } + req_batch = []; + } + } + } + if (req_batch.length > 0) { + await this.get_embeddings_batch(req_batch); + req_batch = []; + processed_since_last_save += req_batch.length; + } + file_embed_input += `: +`; + if (note_contents.length < MAX_EMBED_STRING_LENGTH) { + file_embed_input += note_contents; + } else { + const note_meta_cache = this.app.metadataCache.getFileCache(curr_file); + if (typeof note_meta_cache.headings === "undefined") { + file_embed_input += note_contents.substring(0, MAX_EMBED_STRING_LENGTH); + } else { + let note_headings = ""; + for (let j = 0; j < note_meta_cache.headings.length; j++) { + const heading_level = note_meta_cache.headings[j].level; + const heading_text = note_meta_cache.headings[j].heading; + let md_heading = ""; + for (let k = 0; k < heading_level; k++) { + md_heading += "#"; + } + note_headings += `${md_heading} ${heading_text} +`; + } + file_embed_input += note_headings; + if (file_embed_input.length > MAX_EMBED_STRING_LENGTH) { + file_embed_input = file_embed_input.substring(0, MAX_EMBED_STRING_LENGTH); + } + } + } + const file_hash = md5(file_embed_input.trim()); + const existing_hash = this.smart_vec_lite.get_hash(curr_file_key); + if (existing_hash && file_hash === existing_hash) { + this.update_render_log(blocks, file_embed_input); + return; + } + ; + const existing_blocks = this.smart_vec_lite.get_children(curr_file_key); + let existing_has_all_blocks = true; + if (existing_blocks && Array.isArray(existing_blocks) && blocks.length > 0) { + for (let j = 0; j < blocks.length; j++) { + if (existing_blocks.indexOf(blocks[j]) === -1) { + existing_has_all_blocks = false; + break; + } + } + } + if (existing_has_all_blocks) { + const curr_file_size = curr_file.stat.size; + const prev_file_size = this.smart_vec_lite.get_size(curr_file_key); + if (prev_file_size) { + const file_delta_pct = Math.round(Math.abs(curr_file_size - prev_file_size) / curr_file_size * 100); + if (file_delta_pct < 10) { + this.render_log.skipped_low_delta[curr_file.name] = file_delta_pct + "%"; + this.update_render_log(blocks, file_embed_input); + return; + } + } + } + let meta = { + mtime: curr_file.stat.mtime, + hash: file_hash, + path: curr_file.path, + size: curr_file.stat.size, + children: blocks + }; + req_batch.push([curr_file_key, file_embed_input, meta]); + await this.get_embeddings_batch(req_batch); + if (save) { + await this.save_embeddings_to_file(); + } + } + update_render_log(blocks, file_embed_input) { + if (blocks.length > 0) { + this.render_log.tokens_saved_by_cache += file_embed_input.length / 2; + } else { + this.render_log.tokens_saved_by_cache += file_embed_input.length / 4; + } + } + async get_embeddings_batch(req_batch) { + console.log("get_embeddings_batch"); + if (req_batch.length === 0) + return; + const embed_inputs = req_batch.map((req) => req[1]); + const requestResults = await this.request_embedding_from_input(embed_inputs); + if (!requestResults) { + console.log("failed embedding batch"); + this.render_log.failed_embeddings = [...this.render_log.failed_embeddings, ...req_batch.map((req) => req[2].path)]; + return; + } + if (requestResults) { + this.has_new_embeddings = true; + if (this.settings.log_render) { + if (this.settings.log_render_files) { + this.render_log.files = [...this.render_log.files, ...req_batch.map((req) => req[2].path)]; + } + this.render_log.new_embeddings += req_batch.length; + this.render_log.token_usage += requestResults.usage.total_tokens; + } + for (let i = 0; i < requestResults.data.length; i++) { + const vec = requestResults.data[i].embedding; + const index = requestResults.data[i].index; + if (vec) { + const key = req_batch[index][0]; + const meta = req_batch[index][2]; + this.smart_vec_lite.save_embedding(key, vec, meta); + } + } + } + } + async request_embedding_from_input(embed_input, retries = 0) { + if (embed_input.length === 0) { + console.log("embed_input is empty"); + return null; + } + const usedParams = { + model: "text-embedding-ada-002", + input: embed_input + }; + const reqParams = { + url: `https://api.openai.com/v1/embeddings`, + method: "POST", + body: JSON.stringify(usedParams), + headers: { + "Content-Type": "application/json", + "Authorization": `Bearer ${this.settings.api_key}` + } + }; + let resp; + try { + resp = await (0, Obsidian.request)(reqParams); + return JSON.parse(resp); + } catch (error) { + if (error.status === 429 && retries < 3) { + retries++; + const backoff = Math.pow(retries, 2); + console.log(`retrying request (429) in ${backoff} seconds...`); + await new Promise((r) => setTimeout(r, 1e3 * backoff)); + return await this.request_embedding_from_input(embed_input, retries); + } + console.log(resp); + console.log(error); + return null; + } + } + async test_api_key() { + const embed_input = "This is a test of the OpenAI API."; + const resp = await this.request_embedding_from_input(embed_input); + if (resp && resp.usage) { + console.log("API key is valid"); + return true; + } else { + console.log("API key is invalid"); + return false; + } + } + output_render_log() { + if (this.settings.log_render) { + if (this.render_log.new_embeddings === 0) { + return; + } else { + console.log(JSON.stringify(this.render_log, null, 2)); + } + } + this.render_log = {}; + this.render_log.deleted_embeddings = 0; + this.render_log.exclusions_logs = {}; + this.render_log.failed_embeddings = []; + this.render_log.files = []; + this.render_log.new_embeddings = 0; + this.render_log.skipped_low_delta = {}; + this.render_log.token_usage = 0; + this.render_log.tokens_saved_by_cache = 0; + } + // find connections by most similar to current note by cosine similarity + async find_note_connections(current_note = null) { + const curr_key = md5(current_note.path); + let nearest = []; + if (this.nearest_cache[curr_key]) { + nearest = this.nearest_cache[curr_key]; + } else { + for (let j = 0; j < this.file_exclusions.length; j++) { + if (current_note.path.indexOf(this.file_exclusions[j]) > -1) { + this.log_exclusion(this.file_exclusions[j]); + return "excluded"; + } + } + setTimeout(() => { + this.get_all_embeddings(); + }, 3e3); + if (this.smart_vec_lite.mtime_is_current(curr_key, current_note.stat.mtime)) { + } else { + await this.get_file_embeddings(current_note); + } + const vec = this.smart_vec_lite.get_vec(curr_key); + if (!vec) { + return "Error getting embeddings for: " + current_note.path; + } + nearest = this.smart_vec_lite.nearest(vec, { + skip_key: curr_key, + skip_sections: this.settings.skip_sections + }); + this.nearest_cache[curr_key] = nearest; + } + return nearest; + } + // create render_log object of exlusions with number of times skipped as value + log_exclusion(exclusion) { + this.render_log.exclusions_logs[exclusion] = (this.render_log.exclusions_logs[exclusion] || 0) + 1; + } + block_parser(markdown, file_path) { + if (this.settings.skip_sections) { + return []; + } + const lines = markdown.split("\n"); + let blocks = []; + let currentHeaders = []; + const file_breadcrumbs = file_path.replace(".md", "").replace(/\//g, " > "); + let block = ""; + let block_headings = ""; + let block_path = file_path; + let last_heading_line = 0; + let i = 0; + let block_headings_list = []; + for (i = 0; i < lines.length; i++) { + const line = lines[i]; + if (!line.startsWith("#") || ["#", " "].indexOf(line[1]) < 0) { + if (line === "") + continue; + if (["- ", "- [ ] "].indexOf(line) > -1) + continue; + if (currentHeaders.length === 0) + continue; + block += "\n" + line; + continue; + } + last_heading_line = i; + if (i > 0 && last_heading_line !== i - 1 && block.indexOf("\n") > -1 && this.validate_headings(block_headings)) { + output_block(); + } + const level = line.split("#").length - 1; + currentHeaders = currentHeaders.filter((header) => header.level < level); + currentHeaders.push({ header: line.replace(/#/g, "").trim(), level }); + block = file_breadcrumbs; + block += ": " + currentHeaders.map((header) => header.header).join(" > "); + block_headings = "#" + currentHeaders.map((header) => header.header).join("#"); + if (block_headings_list.indexOf(block_headings) > -1) { + let count = 1; + while (block_headings_list.indexOf(`${block_headings}{${count}}`) > -1) { + count++; + } + block_headings = `${block_headings}{${count}}`; + } + block_headings_list.push(block_headings); + block_path = file_path + block_headings; + } + if (last_heading_line !== i - 1 && block.indexOf("\n") > -1 && this.validate_headings(block_headings)) + output_block(); + blocks = blocks.filter((b) => b.length > 50); + return blocks; + function output_block() { + const breadcrumbs_length = block.indexOf("\n") + 1; + const block_length = block.length - breadcrumbs_length; + if (block.length > MAX_EMBED_STRING_LENGTH) { + block = block.substring(0, MAX_EMBED_STRING_LENGTH); + } + blocks.push({ text: block.trim(), path: block_path, length: block_length }); + } + } + // reverse-retrieve block given path + async block_retriever(path, limits = {}) { + limits = { + lines: null, + chars_per_line: null, + max_chars: null, + ...limits + }; + if (path.indexOf("#") < 0) { + console.log("not a block path: " + path); + return false; + } + let block = []; + let block_headings = path.split("#").slice(1); + let heading_occurrence = 0; + if (block_headings[block_headings.length - 1].indexOf("{") > -1) { + heading_occurrence = parseInt(block_headings[block_headings.length - 1].split("{")[1].replace("}", "")); + block_headings[block_headings.length - 1] = block_headings[block_headings.length - 1].split("{")[0]; + } + let currentHeaders = []; + let occurrence_count = 0; + let begin_line = 0; + let i = 0; + const file_path = path.split("#")[0]; + const file = this.app.vault.getAbstractFileByPath(file_path); + if (!(file instanceof Obsidian.TFile)) { + console.log("not a file: " + file_path); + return false; + } + const file_contents = await this.app.vault.cachedRead(file); + const lines = file_contents.split("\n"); + let is_code = false; + for (i = 0; i < lines.length; i++) { + const line = lines[i]; + if (line.indexOf("```") === 0) { + is_code = !is_code; + } + if (is_code) { + continue; + } + if (["- ", "- [ ] "].indexOf(line) > -1) + continue; + if (!line.startsWith("#") || ["#", " "].indexOf(line[1]) < 0) { + continue; + } + const heading_text = line.replace(/#/g, "").trim(); + const heading_index = block_headings.indexOf(heading_text); + if (heading_index < 0) + continue; + if (currentHeaders.length !== heading_index) + continue; + currentHeaders.push(heading_text); + if (currentHeaders.length === block_headings.length) { + if (heading_occurrence === 0) { + begin_line = i + 1; + break; + } + if (occurrence_count === heading_occurrence) { + begin_line = i + 1; + break; + } + occurrence_count++; + currentHeaders.pop(); + continue; + } + } + if (begin_line === 0) + return false; + is_code = false; + let char_count = 0; + for (i = begin_line; i < lines.length; i++) { + if (typeof line_limit === "number" && block.length > line_limit) { + block.push("..."); + break; + } + let line = lines[i]; + if (line.indexOf("#") === 0 && ["#", " "].indexOf(line[1]) !== -1) { + break; + } + if (limits.max_chars && char_count > limits.max_chars) { + block.push("..."); + break; + } + if (limits.max_chars && line.length + char_count > limits.max_chars) { + const max_new_chars = limits.max_chars - char_count; + line = line.slice(0, max_new_chars) + "..."; + break; + } + if (line.length === 0) + continue; + if (limits.chars_per_line && line.length > limits.chars_per_line) { + line = line.slice(0, limits.chars_per_line) + "..."; + } + if (line.startsWith("```")) { + is_code = !is_code; + continue; + } + if (is_code) { + line = " " + line; + } + block.push(line); + char_count += line.length; + } + if (is_code) { + block.push("```"); + } + return block.join("\n").trim(); + } + // retrieve a file from the vault + async file_retriever(link, limits = {}) { + limits = { + lines: null, + max_chars: null, + chars_per_line: null, + ...limits + }; + const this_file = this.app.vault.getAbstractFileByPath(link); + if (!(this_file instanceof Obsidian.TAbstractFile)) + return false; + const file_content = await this.app.vault.cachedRead(this_file); + const file_lines = file_content.split("\n"); + let first_ten_lines = []; + let is_code = false; + let char_accum = 0; + const line_limit2 = limits.lines || file_lines.length; + for (let i = 0; first_ten_lines.length < line_limit2; i++) { + let line = file_lines[i]; + if (typeof line === "undefined") + break; + if (line.length === 0) + continue; + if (limits.chars_per_line && line.length > limits.chars_per_line) { + line = line.slice(0, limits.chars_per_line) + "..."; + } + if (line === "---") + continue; + if (["- ", "- [ ] "].indexOf(line) > -1) + continue; + if (line.indexOf("```") === 0) { + is_code = !is_code; + continue; + } + if (limits.max_chars && char_accum > limits.max_chars) { + first_ten_lines.push("..."); + break; + } + if (is_code) { + line = " " + line; + } + if (line_is_heading(line)) { + if (first_ten_lines.length > 0 && line_is_heading(first_ten_lines[first_ten_lines.length - 1])) { + first_ten_lines.pop(); + } + } + first_ten_lines.push(line); + char_accum += line.length; + } + for (let i = 0; i < first_ten_lines.length; i++) { + if (line_is_heading(first_ten_lines[i])) { + if (i === first_ten_lines.length - 1) { + first_ten_lines.pop(); + break; + } + first_ten_lines[i] = first_ten_lines[i].replace(/#+/, ""); + first_ten_lines[i] = ` +${first_ten_lines[i]}:`; + } + } + first_ten_lines = first_ten_lines.join("\n"); + return first_ten_lines; + } + // iterate through blocks and skip if block_headings contains this.header_exclusions + validate_headings(block_headings) { + let valid = true; + if (this.header_exclusions.length > 0) { + for (let k = 0; k < this.header_exclusions.length; k++) { + if (block_headings.indexOf(this.header_exclusions[k]) > -1) { + valid = false; + this.log_exclusion("heading: " + this.header_exclusions[k]); + break; + } + } + } + return valid; + } + // render "Smart Connections" text fixed in the bottom right corner + render_brand(container, location = "default") { + if (container === "all") { + const locations = Object.keys(this.sc_branding); + for (let i = 0; i < locations.length; i++) { + this.render_brand(this.sc_branding[locations[i]], locations[i]); + } + return; + } + this.sc_branding[location] = container; + if (this.sc_branding[location].querySelector(".sc-brand")) { + this.sc_branding[location].querySelector(".sc-brand").remove(); + } + const brand_container = this.sc_branding[location].createEl("div", { cls: "sc-brand" }); + Obsidian.setIcon(brand_container, "smart-connections"); + const brand_p = brand_container.createEl("p"); + let text = "Smart Connections"; + let attr = {}; + if (this.update_available) { + text = "Update Available"; + attr = { + style: "font-weight: 700;" + }; + } + brand_p.createEl("a", { + cls: "", + text, + href: "https://github.com/brianpetro/obsidian-smart-connections/discussions", + target: "_blank", + attr + }); + } + // create list of nearest notes + async update_results(container, nearest) { + let list; + if (container.children.length > 1 && container.children[1].classList.contains("sc-list")) { + list = container.children[1]; + } + if (list) { + list.empty(); + } else { + list = container.createEl("div", { cls: "sc-list" }); + } + let search_result_class = "search-result"; + if (!this.settings.expanded_view) + search_result_class += " sc-collapsed"; + if (!this.settings.group_nearest_by_file) { + for (let i = 0; i < nearest.length; i++) { + console.log(this); + if (typeof nearest[i].link === "object") { + const item2 = list.createEl("div", { cls: "search-result" }); + const link2 = item2.createEl("a", { + cls: "search-result-file-title is-clickable", + href: nearest[i].link.path, + title: nearest[i].link.title + }); + link2.innerHTML = this.render_external_link_elm(nearest[i].link); + item2.setAttr("draggable", "true"); + continue; + } + let file_link_text; + const file_similarity_pct = Math.round(nearest[i].similarity * 100) + "%"; + if (this.settings.show_full_path) { + const pcs = nearest[i].link.split("/"); + file_link_text = pcs[pcs.length - 1]; + const path = pcs.slice(0, pcs.length - 1).join("/"); + file_link_text = `${file_similarity_pct} | ${path} | ${file_link_text}`; + } else { + file_link_text = "" + file_similarity_pct + " | " + nearest[i].link.split("/").pop() + ""; + } + if (!this.renderable_file_type(nearest[i].link)) { + const item2 = list.createEl("div", { cls: "search-result" }); + const link2 = item2.createEl("a", { + cls: "search-result-file-title is-clickable", + href: nearest[i].link + }); + link2.innerHTML = file_link_text; + item2.setAttr("draggable", "true"); + this.add_link_listeners(link2, nearest[i], item2); + continue; + } + file_link_text = file_link_text.replace(".md", "").replace(/#/g, " > "); + const item = list.createEl("div", { cls: search_result_class }); + const toggle = item.createEl("span", { cls: "is-clickable" }); + Obsidian.setIcon(toggle, "right-triangle"); + const link = toggle.createEl("a", { + cls: "search-result-file-title", + title: nearest[i].link + }); + link.innerHTML = file_link_text; + this.add_link_listeners(link, nearest[i], item); + toggle.addEventListener("click", (event) => { + let parent = event.target.parentElement; + while (!parent.classList.contains("search-result")) { + parent = parent.parentElement; + } + parent.classList.toggle("sc-collapsed"); + }); + const contents = item.createEl("ul", { cls: "" }); + const contents_container = contents.createEl("li", { + cls: "search-result-file-title is-clickable", + title: nearest[i].link + }); + if (nearest[i].link.indexOf("#") > -1) { + Obsidian.MarkdownRenderer.renderMarkdown(await this.block_retriever(nearest[i].link, { lines: 10, max_chars: 1e3 }), contents_container, nearest[i].link, new Obsidian.Component()); + } else { + const first_ten_lines = await this.file_retriever(nearest[i].link, { lines: 10, max_chars: 1e3 }); + if (!first_ten_lines) + continue; + Obsidian.MarkdownRenderer.renderMarkdown(first_ten_lines, contents_container, nearest[i].link, new Obsidian.Component()); + } + this.add_link_listeners(contents, nearest[i], item); + } + this.render_brand(container, "block"); + return; + } + const nearest_by_file = {}; + for (let i = 0; i < nearest.length; i++) { + const curr = nearest[i]; + const link = curr.link; + if (typeof link === "object") { + nearest_by_file[link.path] = [curr]; + continue; + } + if (link.indexOf("#") > -1) { + const file_path = link.split("#")[0]; + if (!nearest_by_file[file_path]) { + nearest_by_file[file_path] = []; + } + nearest_by_file[file_path].push(nearest[i]); + } else { + if (!nearest_by_file[link]) { + nearest_by_file[link] = []; + } + nearest_by_file[link].unshift(nearest[i]); + } + } + const keys = Object.keys(nearest_by_file); + for (let i = 0; i < keys.length; i++) { + const file = nearest_by_file[keys[i]]; + if (typeof file[0].link === "object") { + const curr = file[0]; + const meta = curr.link; + if (meta.path.startsWith("http")) { + const item2 = list.createEl("div", { cls: "search-result" }); + const link = item2.createEl("a", { + cls: "search-result-file-title is-clickable", + href: meta.path, + title: meta.title + }); + link.innerHTML = this.render_external_link_elm(meta); + item2.setAttr("draggable", "true"); + continue; + } + } + let file_link_text; + const file_similarity_pct = Math.round(file[0].similarity * 100) + "%"; + if (this.settings.show_full_path) { + const pcs = file[0].link.split("/"); + file_link_text = pcs[pcs.length - 1]; + const path = pcs.slice(0, pcs.length - 1).join("/"); + file_link_text = `${path} | ${file_similarity_pct}
${file_link_text}`; + } else { + file_link_text = file[0].link.split("/").pop(); + file_link_text += " | " + file_similarity_pct; + } + if (!this.renderable_file_type(file[0].link)) { + const item2 = list.createEl("div", { cls: "search-result" }); + const file_link2 = item2.createEl("a", { + cls: "search-result-file-title is-clickable", + title: file[0].link + }); + file_link2.innerHTML = file_link_text; + this.add_link_listeners(file_link2, file[0], item2); + continue; + } + file_link_text = file_link_text.replace(".md", "").replace(/#/g, " > "); + const item = list.createEl("div", { cls: search_result_class }); + const toggle = item.createEl("span", { cls: "is-clickable" }); + Obsidian.setIcon(toggle, "right-triangle"); + const file_link = toggle.createEl("a", { + cls: "search-result-file-title", + title: file[0].link + }); + file_link.innerHTML = file_link_text; + this.add_link_listeners(file_link, file[0], toggle); + toggle.addEventListener("click", (event) => { + let parent = event.target; + while (!parent.classList.contains("search-result")) { + parent = parent.parentElement; + } + parent.classList.toggle("sc-collapsed"); + }); + const file_link_list = item.createEl("ul"); + for (let j = 0; j < file.length; j++) { + if (file[j].link.indexOf("#") > -1) { + const block = file[j]; + const block_link = file_link_list.createEl("li", { + cls: "search-result-file-title is-clickable", + title: block.link + }); + if (file.length > 1) { + const block_context = this.render_block_context(block); + const block_similarity_pct = Math.round(block.similarity * 100) + "%"; + block_link.innerHTML = `${block_context} | ${block_similarity_pct}`; + } + const block_container = block_link.createEl("div"); + Obsidian.MarkdownRenderer.renderMarkdown(await this.block_retriever(block.link, { lines: 10, max_chars: 1e3 }), block_container, block.link, new Obsidian.Component()); + this.add_link_listeners(block_link, block, file_link_list); + } else { + const file_link_list2 = item.createEl("ul"); + const block_link = file_link_list2.createEl("li", { + cls: "search-result-file-title is-clickable", + title: file[0].link + }); + const block_container = block_link.createEl("div"); + let first_ten_lines = await this.file_retriever(file[0].link, { lines: 10, max_chars: 1e3 }); + if (!first_ten_lines) + continue; + Obsidian.MarkdownRenderer.renderMarkdown(first_ten_lines, block_container, file[0].link, new Obsidian.Component()); + this.add_link_listeners(block_link, file[0], file_link_list2); + } + } + } + this.render_brand(container, "file"); + } + add_link_listeners(item, curr, list) { + item.addEventListener("click", async (event) => { + await this.open_note(curr, event); + }); + item.setAttr("draggable", "true"); + item.addEventListener("dragstart", (event) => { + const dragManager = this.app.dragManager; + const file_path = curr.link.split("#")[0]; + const file = this.app.metadataCache.getFirstLinkpathDest(file_path, ""); + const dragData = dragManager.dragFile(event, file); + dragManager.onDragStart(event, dragData); + }); + if (curr.link.indexOf("{") > -1) + return; + item.addEventListener("mouseover", (event) => { + this.app.workspace.trigger("hover-link", { + event, + source: SMART_CONNECTIONS_VIEW_TYPE, + hoverParent: list, + targetEl: item, + linktext: curr.link + }); + }); + } + // get target file from link path + // if sub-section is linked, open file and scroll to sub-section + async open_note(curr, event = null) { + let targetFile; + let heading; + if (curr.link.indexOf("#") > -1) { + targetFile = this.app.metadataCache.getFirstLinkpathDest(curr.link.split("#")[0], ""); + const target_file_cache = this.app.metadataCache.getFileCache(targetFile); + let heading_text = curr.link.split("#").pop(); + let occurence = 0; + if (heading_text.indexOf("{") > -1) { + occurence = parseInt(heading_text.split("{")[1].split("}")[0]); + heading_text = heading_text.split("{")[0]; + } + const headings = target_file_cache.headings; + for (let i = 0; i < headings.length; i++) { + if (headings[i].heading === heading_text) { + if (occurence === 0) { + heading = headings[i]; + break; + } + occurence--; + } + } + } else { + targetFile = this.app.metadataCache.getFirstLinkpathDest(curr.link, ""); + } + let leaf; + if (event) { + const mod = Obsidian.Keymap.isModEvent(event); + leaf = this.app.workspace.getLeaf(mod); + } else { + leaf = this.app.workspace.getMostRecentLeaf(); + } + await leaf.openFile(targetFile); + if (heading) { + let { editor } = leaf.view; + const pos = { line: heading.position.start.line, ch: 0 }; + editor.setCursor(pos); + editor.scrollIntoView({ to: pos, from: pos }, true); + } + } + render_block_context(block) { + const block_headings = block.link.split(".md")[1].split("#"); + let block_context = ""; + for (let i = block_headings.length - 1; i >= 0; i--) { + if (block_context.length > 0) { + block_context = ` > ${block_context}`; + } + block_context = block_headings[i] + block_context; + if (block_context.length > 100) { + break; + } + } + if (block_context.startsWith(" > ")) { + block_context = block_context.slice(3); + } + return block_context; + } + renderable_file_type(link) { + return link.indexOf(".md") !== -1 && link.indexOf(".excalidraw") === -1; + } + render_external_link_elm(meta) { + if (meta.source) { + if (meta.source === "Gmail") + meta.source = "\u{1F4E7} Gmail"; + return `${meta.source}
${meta.title}`; + } + let domain = meta.path.replace(/(^\w+:|^)\/\//, ""); + domain = domain.split("/")[0]; + return `\u{1F310} ${domain}
${meta.title}`; + } + // get all folders + async get_all_folders() { + if (!this.folders || this.folders.length === 0) { + this.folders = await this.get_folders(); + } + return this.folders; + } + // get folders, traverse non-hidden sub-folders + async get_folders(path = "/") { + let folders = (await this.app.vault.adapter.list(path)).folders; + let folder_list = []; + for (let i = 0; i < folders.length; i++) { + if (folders[i].startsWith(".")) + continue; + folder_list.push(folders[i]); + folder_list = folder_list.concat(await this.get_folders(folders[i] + "/")); + } + return folder_list; + } + async sync_notes() { + if (!this.settings.license_key) { + new Obsidian.Notice("Smart Connections: Supporter license key is required to sync notes to the ChatGPT Plugin server."); + return; + } + console.log("syncing notes"); + const files = this.app.vault.getMarkdownFiles().filter((file) => { + for (let i = 0; i < this.file_exclusions.length; i++) { + if (file.path.indexOf(this.file_exclusions[i]) > -1) { + return false; + } + } + return true; + }); + const notes = await this.build_notes_object(files); + console.log("object built"); + await this.app.vault.adapter.write(".smart-connections/notes.json", JSON.stringify(notes, null, 2)); + console.log("notes saved"); + console.log(this.settings.license_key); + const response = await (0, Obsidian.requestUrl)({ + url: "https://sync.smartconnections.app/sync", + method: "POST", + headers: { + "Content-Type": "application/json" + }, + contentType: "application/json", + body: JSON.stringify({ + license_key: this.settings.license_key, + notes + }) + }); + console.log(response); + } + async build_notes_object(files) { + let output = {}; + for (let i = 0; i < files.length; i++) { + let file = files[i]; + let parts = file.path.split("/"); + let current = output; + for (let ii = 0; ii < parts.length; ii++) { + let part = parts[ii]; + if (ii === parts.length - 1) { + current[part] = await this.app.vault.cachedRead(file); + } else { + if (!current[part]) { + current[part] = {}; + } + current = current[part]; + } + } + } + return output; + } +}; +var SMART_CONNECTIONS_VIEW_TYPE = "smart-connections-view"; +var SmartConnectionsView = class extends Obsidian.ItemView { + constructor(leaf, plugin) { + super(leaf); + this.plugin = plugin; + this.nearest = null; + this.load_wait = null; + } + getViewType() { + return SMART_CONNECTIONS_VIEW_TYPE; + } + getDisplayText() { + return "Smart Connections Files"; + } + getIcon() { + return "smart-connections"; + } + set_message(message) { + const container = this.containerEl.children[1]; + container.empty(); + this.initiate_top_bar(container); + if (Array.isArray(message)) { + for (let i = 0; i < message.length; i++) { + container.createEl("p", { cls: "sc_message", text: message[i] }); + } + } else { + container.createEl("p", { cls: "sc_message", text: message }); + } + } + render_link_text(link, show_full_path = false) { + if (!show_full_path) { + link = link.split("/").pop(); + } + if (link.indexOf("#") > -1) { + link = link.split(".md"); + link[0] = `${link[0]}
`; + link = link.join(""); + link = link.replace(/\#/g, " \xBB "); + } else { + link = link.replace(".md", ""); + } + return link; + } + set_nearest(nearest, nearest_context = null, results_only = false) { + const container = this.containerEl.children[1]; + if (!results_only) { + container.empty(); + this.initiate_top_bar(container, nearest_context); + } + this.plugin.update_results(container, nearest); + } + initiate_top_bar(container, nearest_context = null) { + let top_bar; + if (container.children.length > 0 && container.children[0].classList.contains("sc-top-bar")) { + top_bar = container.children[0]; + top_bar.empty(); + } else { + top_bar = container.createEl("div", { cls: "sc-top-bar" }); + } + if (nearest_context) { + top_bar.createEl("p", { cls: "sc-context", text: nearest_context }); + } + const chat_button = top_bar.createEl("button", { cls: "sc-chat-button" }); + Obsidian.setIcon(chat_button, "message-square"); + chat_button.addEventListener("click", () => { + this.plugin.open_chat(); + }); + const search_button = top_bar.createEl("button", { cls: "sc-search-button" }); + Obsidian.setIcon(search_button, "search"); + search_button.addEventListener("click", () => { + top_bar.empty(); + const search_container = top_bar.createEl("div", { cls: "search-input-container" }); + const input = search_container.createEl("input", { + cls: "sc-search-input", + type: "search", + placeholder: "Type to start search..." + }); + input.focus(); + input.addEventListener("keydown", (event) => { + if (event.key === "Escape") { + this.clear_auto_searcher(); + this.initiate_top_bar(container, nearest_context); + } + }); + input.addEventListener("keyup", (event) => { + this.clear_auto_searcher(); + const search_term = input.value; + if (event.key === "Enter" && search_term !== "") { + this.search(search_term); + } else if (search_term !== "") { + clearTimeout(this.search_timeout); + this.search_timeout = setTimeout(() => { + this.search(search_term, true); + }, 700); + } + }); + }); + } + // render buttons: "create" and "retry" for loading embeddings.json file + render_embeddings_buttons() { + const container = this.containerEl.children[1]; + container.empty(); + container.createEl("h2", { cls: "scHeading", text: "Embeddings file not found" }); + const button_div = container.createEl("div", { cls: "scButtonDiv" }); + const create_button = button_div.createEl("button", { cls: "scButton", text: "Create embeddings.json" }); + button_div.createEl("p", { cls: "scButtonNote", text: "Warning: Creating embeddings.json file will trigger bulk embedding and may take a while" }); + const retry_button = button_div.createEl("button", { cls: "scButton", text: "Retry" }); + button_div.createEl("p", { cls: "scButtonNote", text: "If embeddings.json file already exists, click 'Retry' to load it" }); + create_button.addEventListener("click", async (event) => { + await this.plugin.smart_vec_lite.init_embeddings_file(); + await this.render_connections(); + }); + retry_button.addEventListener("click", async (event) => { + console.log("retrying to load embeddings.json file"); + await this.plugin.init_vecs(); + await this.render_connections(); + }); + } + async onOpen() { + const container = this.containerEl.children[1]; + container.empty(); + container.createEl("p", { cls: "scPlaceholder", text: "Open a note to find connections." }); + this.plugin.registerEvent(this.app.workspace.on("file-open", (file) => { + if (!file) { + return; + } + if (SUPPORTED_FILE_TYPES.indexOf(file.extension) === -1) { + return this.set_message([ + "File: " + file.name, + "Unsupported file type (Supported: " + SUPPORTED_FILE_TYPES.join(", ") + ")" + ]); + } + if (this.load_wait) { + clearTimeout(this.load_wait); + } + this.load_wait = setTimeout(() => { + this.render_connections(file); + this.load_wait = null; + }, 1e3); + })); + this.app.workspace.registerHoverLinkSource(SMART_CONNECTIONS_VIEW_TYPE, { + display: "Smart Connections Files", + defaultMod: true + }); + this.app.workspace.registerHoverLinkSource(SMART_CONNECTIONS_CHAT_VIEW_TYPE, { + display: "Smart Chat Links", + defaultMod: true + }); + this.app.workspace.onLayoutReady(this.initialize.bind(this)); + } + async initialize() { + this.set_message("Loading embeddings file..."); + const vecs_intiated = await this.plugin.init_vecs(); + if (vecs_intiated) { + this.set_message("Embeddings file loaded."); + await this.render_connections(); + } else { + this.render_embeddings_buttons(); + } + this.api = new SmartConnectionsViewApi(this.app, this.plugin, this); + (window["SmartConnectionsViewApi"] = this.api) && this.register(() => delete window["SmartConnectionsViewApi"]); + } + async onClose() { + console.log("closing smart connections view"); + this.app.workspace.unregisterHoverLinkSource(SMART_CONNECTIONS_VIEW_TYPE); + this.plugin.view = null; + } + async render_connections(context = null) { + console.log("rendering connections"); + if (!this.plugin.settings.api_key) { + this.set_message("An OpenAI API key is required to make Smart Connections"); + return; + } + if (!this.plugin.embeddings_loaded) { + await this.plugin.init_vecs(); + } + if (!this.plugin.embeddings_loaded) { + console.log("embeddings files still not loaded or yet to be created"); + this.render_embeddings_buttons(); + return; + } + this.set_message("Making Smart Connections..."); + if (typeof context === "string") { + const highlighted_text = context; + await this.search(highlighted_text); + return; + } + this.nearest = null; + this.interval_count = 0; + this.rendering = false; + this.file = context; + if (this.interval) { + clearInterval(this.interval); + this.interval = null; + } + this.interval = setInterval(() => { + if (!this.rendering) { + if (this.file instanceof Obsidian.TFile) { + this.rendering = true; + this.render_note_connections(this.file); + } else { + this.file = this.app.workspace.getActiveFile(); + if (!this.file && this.count > 1) { + clearInterval(this.interval); + this.set_message("No active file"); + return; + } + } + } else { + if (this.nearest) { + clearInterval(this.interval); + if (typeof this.nearest === "string") { + this.set_message(this.nearest); + } else { + this.set_nearest(this.nearest, "File: " + this.file.name); + } + if (this.plugin.render_log.failed_embeddings.length > 0) { + this.plugin.save_failed_embeddings(); + } + this.plugin.output_render_log(); + return; + } else { + this.interval_count++; + this.set_message("Making Smart Connections..." + this.interval_count); + } + } + }, 10); + } + async render_note_connections(file) { + this.nearest = await this.plugin.find_note_connections(file); + } + clear_auto_searcher() { + if (this.search_timeout) { + clearTimeout(this.search_timeout); + this.search_timeout = null; + } + } + async search(search_text, results_only = false) { + const nearest = await this.plugin.api.search(search_text); + const nearest_context = `Selection: "${search_text.length > 100 ? search_text.substring(0, 100) + "..." : search_text}"`; + this.set_nearest(nearest, nearest_context, results_only); + } +}; +var SmartConnectionsViewApi = class { + constructor(app, plugin, view) { + this.app = app; + this.plugin = plugin; + this.view = view; + } + async search(search_text) { + return await this.plugin.api.search(search_text); + } + // trigger reload of embeddings file + async reload_embeddings_file() { + await this.plugin.init_vecs(); + await this.view.render_connections(); + } + async init_vecs() { + this.smart_vec_lite = new VecLite({ + folder_path: ".smart-connections", + exists_adapter: this.app.vault.adapter.exists.bind( + this.app.vault.adapter + ), + mkdir_adapter: this.app.vault.adapter.mkdir.bind(this.app.vault.adapter), + read_adapter: this.app.vault.adapter.read.bind(this.app.vault.adapter), + rename_adapter: this.app.vault.adapter.rename.bind( + this.app.vault.adapter + ), + stat_adapter: this.app.vault.adapter.stat.bind(this.app.vault.adapter), + write_adapter: this.app.vault.adapter.write.bind(this.app.vault.adapter) + }); + this.embeddings_loaded = await this.smart_vec_lite.load(); + return this.embeddings_loaded; + } +}; +var ScSearchApi = class { + constructor(app, plugin) { + this.app = app; + this.plugin = plugin; + } + async search(search_text, filter = {}) { + filter = { + skip_sections: this.plugin.settings.skip_sections, + ...filter + }; + let nearest = []; + const resp = await this.plugin.request_embedding_from_input(search_text); + if (resp && resp.data && resp.data[0] && resp.data[0].embedding) { + nearest = this.plugin.smart_vec_lite.nearest(resp.data[0].embedding, filter); + } else { + new Obsidian.Notice("Smart Connections: Error getting embedding"); + } + return nearest; + } +}; +var SmartConnectionsSettingsTab = class extends Obsidian.PluginSettingTab { + constructor(app, plugin) { + super(app, plugin); + this.plugin = plugin; + } + display() { + const { + containerEl + } = this; + containerEl.empty(); + containerEl.createEl("h2", { + text: "Supporter Settings" + }); + containerEl.createEl("p", { + text: 'As a Smart Connections "Supporter", fast-track your PKM journey with priority perks and pioneering innovations.' + }); + const supporter_benefits_list = containerEl.createEl("ul"); + supporter_benefits_list.createEl("li", { + text: "Enjoy swift, top-priority support." + }); + supporter_benefits_list.createEl("li", { + text: "Gain early access to version 2 (includes local embedding model)." + }); + supporter_benefits_list.createEl("li", { + text: "Stay informed and engaged with exclusive supporter-only communications." + }); + new Obsidian.Setting(containerEl).setName("Supporter License Key").setDesc("Note: this is not required to use Smart Connections.").addText((text) => text.setPlaceholder("Enter your license_key").setValue(this.plugin.settings.license_key).onChange(async (value) => { + this.plugin.settings.license_key = value.trim(); + await this.plugin.saveSettings(true); + })); + new Obsidian.Setting(containerEl).setName("Get v2").setDesc("Get v2 (warning: very early beta release, likely to crash, please send issues directly to the supporter email for quick response)").addButton((button) => button.setButtonText("Get v2 (unstable)").onClick(async () => { + await this.plugin.update_to_v2(); + })); + new Obsidian.Setting(containerEl).setName("Sync Notes").setDesc("Make notes available via the Smart Connections ChatGPT Plugin. Respects exclusion settings configured below.").addButton((button) => button.setButtonText("Sync Notes").onClick(async () => { + await this.plugin.sync_notes(); + })); + new Obsidian.Setting(containerEl).setName("Become a Supporter").setDesc("Become a Supporter").addButton((button) => button.setButtonText("Become a Supporter").onClick(async () => { + const payment_pages = [ + "https://buy.stripe.com/9AQ5kO5QnbAWgGAbIY", + "https://buy.stripe.com/9AQ7sWemT48u1LGcN4" + ]; + if (!this.plugin.payment_page_index) { + this.plugin.payment_page_index = Math.round(Math.random()); + } + window.open(payment_pages[this.plugin.payment_page_index]); + })); + containerEl.createEl("h2", { + text: "OpenAI Settings" + }); + new Obsidian.Setting(containerEl).setName("OpenAI API Key").setDesc("Required: an OpenAI API key is currently required to use Smart Connections.").addText((text) => text.setPlaceholder("Enter your api_key").setValue(this.plugin.settings.api_key).onChange(async (value) => { + this.plugin.settings.api_key = value.trim(); + await this.plugin.saveSettings(true); + })); + new Obsidian.Setting(containerEl).setName("Test API Key").setDesc("Test API Key").addButton((button) => button.setButtonText("Test API Key").onClick(async () => { + const resp = await this.plugin.test_api_key(); + if (resp) { + new Obsidian.Notice("Smart Connections: API key is valid"); + } else { + new Obsidian.Notice("Smart Connections: API key is not working as expected!"); + } + })); + new Obsidian.Setting(containerEl).setName("Smart Chat Model").setDesc("Select a model to use with Smart Chat.").addDropdown((dropdown) => { + dropdown.addOption("gpt-3.5-turbo-16k", "gpt-3.5-turbo-16k"); + dropdown.addOption("gpt-4", "gpt-4 (limited access, 8k)"); + dropdown.addOption("gpt-3.5-turbo", "gpt-3.5-turbo (4k)"); + dropdown.addOption("gpt-4-1106-preview", "gpt-4-turbo (128k)"); + dropdown.onChange(async (value) => { + this.plugin.settings.smart_chat_model = value; + await this.plugin.saveSettings(); + }); + dropdown.setValue(this.plugin.settings.smart_chat_model); + }); + new Obsidian.Setting(containerEl).setName("Default Language").setDesc("Default language to use for Smart Chat. Changes which self-referential pronouns will trigger lookup of your notes.").addDropdown((dropdown) => { + const languages = Object.keys(SMART_TRANSLATION); + for (let i = 0; i < languages.length; i++) { + dropdown.addOption(languages[i], languages[i]); + } + dropdown.onChange(async (value) => { + this.plugin.settings.language = value; + await this.plugin.saveSettings(); + self_ref_pronouns_list.setText(this.get_self_ref_list()); + const chat_view = this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE).length > 0 ? this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE)[0].view : null; + if (chat_view) { + chat_view.new_chat(); + } + }); + dropdown.setValue(this.plugin.settings.language); + }); + const self_ref_pronouns_list = containerEl.createEl("span", { + text: this.get_self_ref_list() + }); + new Obsidian.Setting(containerEl).setName("Cut off frontmatter").setDesc("Cut off frontmatter in the prompt to gain characters in reply generation").addToggle((toggle) => { + toggle.setValue(this.plugin.settings.cut_off_frontmatter).onChange(async (value) => { + this.plugin.settings.cut_off_frontmatter = value; + await this.plugin.saveSettings(); + }); + }); + containerEl.createEl("h2", { + text: "Exclusions" + }); + new Obsidian.Setting(containerEl).setName("file_exclusions").setDesc("'Excluded file' matchers separated by a comma.").addText((text) => text.setPlaceholder("drawings,prompts/logs").setValue(this.plugin.settings.file_exclusions).onChange(async (value) => { + this.plugin.settings.file_exclusions = value; + await this.plugin.saveSettings(); + })); + new Obsidian.Setting(containerEl).setName("folder_exclusions").setDesc("'Excluded folder' matchers separated by a comma.").addText((text) => text.setPlaceholder("drawings,prompts/logs").setValue(this.plugin.settings.folder_exclusions).onChange(async (value) => { + this.plugin.settings.folder_exclusions = value; + await this.plugin.saveSettings(); + })); + new Obsidian.Setting(containerEl).setName("path_only").setDesc("'Path only' matchers separated by a comma.").addText((text) => text.setPlaceholder("drawings,prompts/logs").setValue(this.plugin.settings.path_only).onChange(async (value) => { + this.plugin.settings.path_only = value; + await this.plugin.saveSettings(); + })); + new Obsidian.Setting(containerEl).setName("header_exclusions").setDesc("'Excluded header' matchers separated by a comma. Works for 'blocks' only.").addText((text) => text.setPlaceholder("drawings,prompts/logs").setValue(this.plugin.settings.header_exclusions).onChange(async (value) => { + this.plugin.settings.header_exclusions = value; + await this.plugin.saveSettings(); + })); + containerEl.createEl("h2", { + text: "Display" + }); + new Obsidian.Setting(containerEl).setName("show_full_path").setDesc("Show full path in view.").addToggle((toggle) => toggle.setValue(this.plugin.settings.show_full_path).onChange(async (value) => { + this.plugin.settings.show_full_path = value; + await this.plugin.saveSettings(true); + })); + new Obsidian.Setting(containerEl).setName("expanded_view").setDesc("Expanded view by default.").addToggle((toggle) => toggle.setValue(this.plugin.settings.expanded_view).onChange(async (value) => { + this.plugin.settings.expanded_view = value; + await this.plugin.saveSettings(true); + })); + new Obsidian.Setting(containerEl).setName("group_nearest_by_file").setDesc("Group nearest by file.").addToggle((toggle) => toggle.setValue(this.plugin.settings.group_nearest_by_file).onChange(async (value) => { + this.plugin.settings.group_nearest_by_file = value; + await this.plugin.saveSettings(true); + })); + new Obsidian.Setting(containerEl).setName("view_open").setDesc("Open view on Obsidian startup.").addToggle((toggle) => toggle.setValue(this.plugin.settings.view_open).onChange(async (value) => { + this.plugin.settings.view_open = value; + await this.plugin.saveSettings(true); + })); + new Obsidian.Setting(containerEl).setName("chat_open").setDesc("Open view on Obsidian startup.").addToggle((toggle) => toggle.setValue(this.plugin.settings.chat_open).onChange(async (value) => { + this.plugin.settings.chat_open = value; + await this.plugin.saveSettings(true); + })); + new Obsidian.Setting(containerEl).setName("open_in_big_view").setDesc("Open in big view or small view.").addDropdown((dropdown) => { + dropdown.addOption(false, "Right pane (small)"); + dropdown.addOption(true, "Main pane (big)"); + dropdown.setValue(this.plugin.settings.open_in_big_view); + dropdown.onChange(async (value) => { + this.plugin.settings.open_in_big_view = JSON.parse(value); + await this.plugin.saveSettings(true); + this.plugin.open_chat(); + }); + }); + containerEl.createEl("h2", { + text: "Advanced" + }); + new Obsidian.Setting(containerEl).setName("log_render").setDesc("Log render details to console (includes token_usage).").addToggle((toggle) => toggle.setValue(this.plugin.settings.log_render).onChange(async (value) => { + this.plugin.settings.log_render = value; + await this.plugin.saveSettings(true); + })); + new Obsidian.Setting(containerEl).setName("log_render_files").setDesc("Log embedded objects paths with log render (for debugging).").addToggle((toggle) => toggle.setValue(this.plugin.settings.log_render_files).onChange(async (value) => { + this.plugin.settings.log_render_files = value; + await this.plugin.saveSettings(true); + })); + new Obsidian.Setting(containerEl).setName("skip_sections").setDesc("Skips making connections to specific sections within notes. Warning: reduces usefulness for large files and requires 'Force Refresh' for sections to work in the future.").addToggle((toggle) => toggle.setValue(this.plugin.settings.skip_sections).onChange(async (value) => { + this.plugin.settings.skip_sections = value; + await this.plugin.saveSettings(true); + })); + containerEl.createEl("h3", { + text: "Test File Writing" + }); + containerEl.createEl("h3", { + text: "Manual Save" + }); + let manual_save_results = containerEl.createEl("div"); + new Obsidian.Setting(containerEl).setName("manual_save").setDesc("Save current embeddings").addButton((button) => button.setButtonText("Manual Save").onClick(async () => { + if (confirm("Are you sure you want to save your current embeddings?")) { + try { + await this.plugin.save_embeddings_to_file(true); + manual_save_results.innerHTML = "Embeddings saved successfully."; + } catch (e) { + manual_save_results.innerHTML = "Embeddings failed to save. Error: " + e; + } + } + })); + containerEl.createEl("h3", { + text: "Previously failed files" + }); + let failed_list = containerEl.createEl("div"); + this.draw_failed_files_list(failed_list); + containerEl.createEl("h3", { + text: "Force Refresh" + }); + new Obsidian.Setting(containerEl).setName("force_refresh").setDesc("WARNING: DO NOT use unless you know what you are doing! This will delete all of your current embeddings from OpenAI and trigger reprocessing of your entire vault!").addButton((button) => button.setButtonText("Force Refresh").onClick(async () => { + if (confirm("Are you sure you want to Force Refresh? By clicking yes you confirm that you understand the consequences of this action.")) { + await this.plugin.force_refresh_embeddings_file(); + } + })); + } + get_self_ref_list() { + return "Current: " + SMART_TRANSLATION[this.plugin.settings.language].pronous.join(", "); + } + draw_failed_files_list(failed_list) { + failed_list.empty(); + if (this.plugin.settings.failed_files.length > 0) { + failed_list.createEl("p", { + text: "The following files failed to process and will be skipped until manually retried." + }); + let list = failed_list.createEl("ul"); + for (let failed_file of this.plugin.settings.failed_files) { + list.createEl("li", { + text: failed_file + }); + } + new Obsidian.Setting(failed_list).setName("retry_failed_files").setDesc("Retry failed files only").addButton((button) => button.setButtonText("Retry failed files only").onClick(async () => { + failed_list.empty(); + failed_list.createEl("p", { + text: "Retrying failed files..." + }); + await this.plugin.retry_failed_files(); + this.draw_failed_files_list(failed_list); + })); + } else { + failed_list.createEl("p", { + text: "No failed files" + }); + } + } +}; +function line_is_heading(line) { + return line.indexOf("#") === 0 && ["#", " "].indexOf(line[1]) !== -1; +} +var SMART_CONNECTIONS_CHAT_VIEW_TYPE = "smart-connections-chat-view"; +var SmartConnectionsChatView = class extends Obsidian.ItemView { + constructor(leaf, plugin) { + super(leaf); + this.plugin = plugin; + this.active_elm = null; + this.active_stream = null; + this.brackets_ct = 0; + this.chat = null; + this.chat_box = null; + this.chat_container = null; + this.current_chat_ml = []; + this.files = []; + this.last_from = null; + this.message_container = null; + this.prevent_input = false; + } + getDisplayText() { + return "Smart Connections Chat"; + } + getIcon() { + return "message-square"; + } + getViewType() { + return SMART_CONNECTIONS_CHAT_VIEW_TYPE; + } + onOpen() { + this.new_chat(); + this.plugin.get_all_folders(); + } + onClose() { + this.chat.save_chat(); + this.app.workspace.unregisterHoverLinkSource(SMART_CONNECTIONS_CHAT_VIEW_TYPE); + } + render_chat() { + this.containerEl.empty(); + this.chat_container = this.containerEl.createDiv("sc-chat-container"); + this.render_top_bar(); + this.render_chat_box(); + this.render_chat_input(); + this.plugin.render_brand(this.containerEl, "chat"); + } + // render plus sign for clear button + render_top_bar() { + let top_bar_container = this.chat_container.createDiv("sc-top-bar-container"); + let chat_name = this.chat.name(); + let chat_name_input = top_bar_container.createEl("input", { + attr: { + type: "text", + value: chat_name + }, + cls: "sc-chat-name-input" + }); + chat_name_input.addEventListener("change", this.rename_chat.bind(this)); + let smart_view_btn = this.create_top_bar_button(top_bar_container, "Smart View", "smart-connections"); + smart_view_btn.addEventListener("click", this.open_smart_view.bind(this)); + let save_btn = this.create_top_bar_button(top_bar_container, "Save Chat", "save"); + save_btn.addEventListener("click", this.save_chat.bind(this)); + let history_btn = this.create_top_bar_button(top_bar_container, "Chat History", "history"); + history_btn.addEventListener("click", this.open_chat_history.bind(this)); + const new_chat_btn = this.create_top_bar_button(top_bar_container, "New Chat", "plus"); + new_chat_btn.addEventListener("click", this.new_chat.bind(this)); + } + async open_chat_history() { + const folder = await this.app.vault.adapter.list(".smart-connections/chats"); + this.files = folder.files.map((file) => { + return file.replace(".smart-connections/chats/", "").replace(".json", ""); + }); + if (!this.modal) + this.modal = new SmartConnectionsChatHistoryModal(this.app, this); + this.modal.open(); + } + create_top_bar_button(top_bar_container, title, icon = null) { + let btn = top_bar_container.createEl("button", { + attr: { + title + } + }); + if (icon) { + Obsidian.setIcon(btn, icon); + } else { + btn.innerHTML = title; + } + return btn; + } + // render new chat + new_chat() { + this.clear_chat(); + this.render_chat(); + this.new_messsage_bubble("assistant"); + this.active_elm.innerHTML = "

" + SMART_TRANSLATION[this.plugin.settings.language].initial_message + "

"; + } + // open a chat from the chat history modal + async open_chat(chat_id) { + this.clear_chat(); + await this.chat.load_chat(chat_id); + this.render_chat(); + for (let i = 0; i < this.chat.chat_ml.length; i++) { + await this.render_message(this.chat.chat_ml[i].content, this.chat.chat_ml[i].role); + } + } + // clear current chat state + clear_chat() { + if (this.chat) { + this.chat.save_chat(); + } + this.chat = new SmartConnectionsChatModel(this.plugin); + if (this.dotdotdot_interval) { + clearInterval(this.dotdotdot_interval); + } + this.current_chat_ml = []; + this.end_stream(); + } + rename_chat(event) { + let new_chat_name = event.target.value; + this.chat.rename_chat(new_chat_name); + } + // save current chat + save_chat() { + this.chat.save_chat(); + new Obsidian.Notice("[Smart Connections] Chat saved"); + } + open_smart_view() { + this.plugin.open_view(); + } + // render chat messages container + render_chat_box() { + this.chat_box = this.chat_container.createDiv("sc-chat-box"); + this.message_container = this.chat_box.createDiv("sc-message-container"); + } + // open file suggestion modal + open_file_suggestion_modal() { + if (!this.file_selector) + this.file_selector = new SmartConnectionsFileSelectModal(this.app, this); + this.file_selector.open(); + } + // open folder suggestion modal + async open_folder_suggestion_modal() { + if (!this.folder_selector) { + this.folder_selector = new SmartConnectionsFolderSelectModal(this.app, this); + } + this.folder_selector.open(); + } + // insert_selection from file suggestion modal + insert_selection(insert_text) { + let caret_pos = this.textarea.selectionStart; + let text_before = this.textarea.value.substring(0, caret_pos); + let text_after = this.textarea.value.substring(caret_pos, this.textarea.value.length); + this.textarea.value = text_before + insert_text + text_after; + this.textarea.selectionStart = caret_pos + insert_text.length; + this.textarea.selectionEnd = caret_pos + insert_text.length; + this.textarea.focus(); + } + // render chat textarea and button + render_chat_input() { + let chat_input = this.chat_container.createDiv("sc-chat-form"); + this.textarea = chat_input.createEl("textarea", { + cls: "sc-chat-input", + attr: { + placeholder: SMART_TRANSLATION[this.plugin.settings.language].try_placeholder + } + }); + chat_input.addEventListener("keyup", (e) => { + if (["[", "/"].indexOf(e.key) === -1) + return; + const caret_pos = this.textarea.selectionStart; + if (e.key === "[") { + if (this.textarea.value[caret_pos - 2] === "[") { + this.open_file_suggestion_modal(); + return; + } + } else { + this.brackets_ct = 0; + } + if (e.key === "/") { + if (this.textarea.value.length === 1 || this.textarea.value[caret_pos - 2] === " ") { + this.open_folder_suggestion_modal(); + return; + } + } + }); + chat_input.addEventListener("keydown", (e) => { + if (e.key === "Enter" && e.shiftKey) { + e.preventDefault(); + if (this.prevent_input) { + console.log("wait until current response is finished"); + new Obsidian.Notice("[Smart Connections] Wait until current response is finished"); + return; + } + let user_input = this.textarea.value; + this.textarea.value = ""; + this.initialize_response(user_input); + } + this.textarea.style.height = "auto"; + this.textarea.style.height = this.textarea.scrollHeight + "px"; + }); + let button_container = chat_input.createDiv("sc-button-container"); + let abort_button = button_container.createEl("span", { attr: { id: "sc-abort-button", style: "display: none;" } }); + Obsidian.setIcon(abort_button, "square"); + abort_button.addEventListener("click", () => { + this.end_stream(); + }); + let button = button_container.createEl("button", { attr: { id: "sc-send-button" }, cls: "send-button" }); + button.innerHTML = "Send"; + button.addEventListener("click", () => { + if (this.prevent_input) { + console.log("wait until current response is finished"); + new Obsidian.Notice("Wait until current response is finished"); + return; + } + let user_input = this.textarea.value; + this.textarea.value = ""; + this.initialize_response(user_input); + }); + } + async initialize_response(user_input) { + this.set_streaming_ux(); + await this.render_message(user_input, "user"); + this.chat.new_message_in_thread({ + role: "user", + content: user_input + }); + await this.render_dotdotdot(); + if (this.chat.contains_internal_link(user_input)) { + this.chat.get_response_with_note_context(user_input, this); + return; + } + if (this.contains_self_referential_keywords(user_input) || this.chat.contains_folder_reference(user_input)) { + const context = await this.get_context_hyde(user_input); + const chatml = [ + { + role: "system", + // content: context_input + content: context + }, + { + role: "user", + content: user_input + } + ]; + this.request_chatgpt_completion({ messages: chatml, temperature: 0 }); + return; + } + this.request_chatgpt_completion(); + } + async render_dotdotdot() { + if (this.dotdotdot_interval) + clearInterval(this.dotdotdot_interval); + await this.render_message("...", "assistant"); + let dots = 0; + this.active_elm.innerHTML = "..."; + this.dotdotdot_interval = setInterval(() => { + dots++; + if (dots > 3) + dots = 1; + this.active_elm.innerHTML = ".".repeat(dots); + }, 500); + } + set_streaming_ux() { + this.prevent_input = true; + if (document.getElementById("sc-send-button")) + document.getElementById("sc-send-button").style.display = "none"; + if (document.getElementById("sc-abort-button")) + document.getElementById("sc-abort-button").style.display = "block"; + } + unset_streaming_ux() { + this.prevent_input = false; + if (document.getElementById("sc-send-button")) + document.getElementById("sc-send-button").style.display = ""; + if (document.getElementById("sc-abort-button")) + document.getElementById("sc-abort-button").style.display = "none"; + } + // check if includes keywords referring to one's own notes + contains_self_referential_keywords(user_input) { + const matches = user_input.match(this.plugin.self_ref_kw_regex); + if (matches) + return true; + return false; + } + // render message + async render_message(message, from = "assistant", append_last = false) { + if (this.dotdotdot_interval) { + clearInterval(this.dotdotdot_interval); + this.dotdotdot_interval = null; + this.active_elm.innerHTML = ""; + } + if (append_last) { + this.current_message_raw += message; + if (message.indexOf("\n") === -1) { + this.active_elm.innerHTML += message; + } else { + this.active_elm.innerHTML = ""; + await Obsidian.MarkdownRenderer.renderMarkdown(this.current_message_raw, this.active_elm, "?no-dataview", new Obsidian.Component()); + } + } else { + this.current_message_raw = ""; + if (this.chat.thread.length === 0 || this.last_from !== from) { + this.new_messsage_bubble(from); + } + this.active_elm.innerHTML = ""; + await Obsidian.MarkdownRenderer.renderMarkdown(message, this.active_elm, "?no-dataview", new Obsidian.Component()); + this.handle_links_in_message(); + this.render_message_action_buttons(message); + } + this.message_container.scrollTop = this.message_container.scrollHeight; + } + render_message_action_buttons(message) { + if (this.chat.context && this.chat.hyd) { + const context_view = this.active_elm.createEl("span", { + cls: "sc-msg-button", + attr: { + title: "Copy context to clipboard" + /* tooltip */ + } + }); + const this_hyd = this.chat.hyd; + Obsidian.setIcon(context_view, "eye"); + context_view.addEventListener("click", () => { + navigator.clipboard.writeText("```smart-connections\n" + this_hyd + "\n```\n"); + new Obsidian.Notice("[Smart Connections] Context code block copied to clipboard"); + }); + } + if (this.chat.context) { + const copy_prompt_button = this.active_elm.createEl("span", { + cls: "sc-msg-button", + attr: { + title: "Copy prompt to clipboard" + /* tooltip */ + } + }); + const this_context = this.chat.context.replace(/\`\`\`/g, " ```").trimLeft(); + Obsidian.setIcon(copy_prompt_button, "files"); + copy_prompt_button.addEventListener("click", () => { + navigator.clipboard.writeText("```prompt-context\n" + this_context + "\n```\n"); + new Obsidian.Notice("[Smart Connections] Context copied to clipboard"); + }); + } + const copy_button = this.active_elm.createEl("span", { + cls: "sc-msg-button", + attr: { + title: "Copy message to clipboard" + /* tooltip */ + } + }); + Obsidian.setIcon(copy_button, "copy"); + copy_button.addEventListener("click", () => { + navigator.clipboard.writeText(message.trimLeft()); + new Obsidian.Notice("[Smart Connections] Message copied to clipboard"); + }); + } + handle_links_in_message() { + const links = this.active_elm.querySelectorAll("a"); + if (links.length > 0) { + for (let i = 0; i < links.length; i++) { + const link = links[i]; + const link_text = link.getAttribute("data-href"); + link.addEventListener("mouseover", (event) => { + this.app.workspace.trigger("hover-link", { + event, + source: SMART_CONNECTIONS_CHAT_VIEW_TYPE, + hoverParent: link.parentElement, + targetEl: link, + // extract link text from a.data-href + linktext: link_text + }); + }); + link.addEventListener("click", (event) => { + const link_tfile = this.app.metadataCache.getFirstLinkpathDest(link_text, "/"); + const mod = Obsidian.Keymap.isModEvent(event); + let leaf = this.app.workspace.getLeaf(mod); + leaf.openFile(link_tfile); + }); + } + } + } + new_messsage_bubble(from) { + let message_el = this.message_container.createDiv(`sc-message ${from}`); + this.active_elm = message_el.createDiv("sc-message-content"); + this.last_from = from; + } + async request_chatgpt_completion(opts = {}) { + const chat_ml = opts.messages || opts.chat_ml || this.chat.prepare_chat_ml(); + console.log("chat_ml", chat_ml); + const max_total_tokens = Math.round(get_max_chars(this.plugin.settings.smart_chat_model) / 4); + console.log("max_total_tokens", max_total_tokens); + const curr_token_est = Math.round(JSON.stringify(chat_ml).length / 3); + console.log("curr_token_est", curr_token_est); + let max_available_tokens = max_total_tokens - curr_token_est; + if (max_available_tokens < 0) + max_available_tokens = 200; + else if (max_available_tokens > 4096) + max_available_tokens = 4096; + console.log("max_available_tokens", max_available_tokens); + opts = { + model: this.plugin.settings.smart_chat_model, + messages: chat_ml, + // max_tokens: 250, + max_tokens: max_available_tokens, + temperature: 0.3, + top_p: 1, + presence_penalty: 0, + frequency_penalty: 0, + stream: true, + stop: null, + n: 1, + // logit_bias: logit_bias, + ...opts + }; + if (opts.stream) { + const full_str = await new Promise((resolve, reject) => { + try { + const url = "https://api.openai.com/v1/chat/completions"; + this.active_stream = new ScStreamer(url, { + headers: { + "Content-Type": "application/json", + Authorization: `Bearer ${this.plugin.settings.api_key}` + }, + method: "POST", + payload: JSON.stringify(opts) + }); + let txt = ""; + this.active_stream.addEventListener("message", (e) => { + if (e.data != "[DONE]") { + const payload = JSON.parse(e.data); + const text = payload.choices[0].delta.content; + if (!text) { + return; + } + txt += text; + this.render_message(text, "assistant", true); + } else { + this.end_stream(); + resolve(txt); + } + }); + this.active_stream.addEventListener("readystatechange", (e) => { + if (e.readyState >= 2) { + console.log("ReadyState: " + e.readyState); + } + }); + this.active_stream.addEventListener("error", (e) => { + console.error(e); + new Obsidian.Notice("Smart Connections Error Streaming Response. See console for details."); + this.render_message("*API Error. See console logs for details.*", "assistant"); + this.end_stream(); + reject(e); + }); + this.active_stream.stream(); + } catch (err) { + console.error(err); + new Obsidian.Notice("Smart Connections Error Streaming Response. See console for details."); + this.end_stream(); + reject(err); + } + }); + await this.render_message(full_str, "assistant"); + this.chat.new_message_in_thread({ + role: "assistant", + content: full_str + }); + return; + } else { + try { + const response = await (0, Obsidian.requestUrl)({ + url: `https://api.openai.com/v1/chat/completions`, + method: "POST", + headers: { + Authorization: `Bearer ${this.plugin.settings.api_key}`, + "Content-Type": "application/json" + }, + contentType: "application/json", + body: JSON.stringify(opts), + throw: false + }); + return JSON.parse(response.text).choices[0].message.content; + } catch (err) { + new Obsidian.Notice(`Smart Connections API Error :: ${err}`); + } + } + } + end_stream() { + if (this.active_stream) { + this.active_stream.close(); + this.active_stream = null; + } + this.unset_streaming_ux(); + if (this.dotdotdot_interval) { + clearInterval(this.dotdotdot_interval); + this.dotdotdot_interval = null; + this.active_elm.parentElement.remove(); + this.active_elm = null; + } + } + async get_context_hyde(user_input) { + this.chat.reset_context(); + const hyd_input = `Anticipate what the user is seeking. Respond in the form of a hypothetical note written by the user. The note may contain statements as paragraphs, lists, or checklists in markdown format with no headings. Please respond with one hypothetical note and abstain from any other commentary. Use the format: PARENT FOLDER NAME > CHILD FOLDER NAME > FILE NAME > HEADING 1 > HEADING 2 > HEADING 3: HYPOTHETICAL NOTE CONTENTS.`; + const chatml = [ + { + role: "system", + content: hyd_input + }, + { + role: "user", + content: user_input + } + ]; + const hyd = await this.request_chatgpt_completion({ + messages: chatml, + stream: false, + temperature: 0, + max_tokens: 137 + }); + this.chat.hyd = hyd; + let filter = {}; + if (this.chat.contains_folder_reference(user_input)) { + const folder_refs = this.chat.get_folder_references(user_input); + if (folder_refs) { + filter = { + path_begins_with: folder_refs + }; + } + } + let nearest = await this.plugin.api.search(hyd, filter); + console.log("nearest", nearest.length); + nearest = this.get_nearest_until_next_dev_exceeds_std_dev(nearest); + console.log("nearest after std dev slice", nearest.length); + nearest = this.sort_by_len_adjusted_similarity(nearest); + return await this.get_context_for_prompt(nearest); + } + sort_by_len_adjusted_similarity(nearest) { + nearest = nearest.sort((a, b) => { + const a_score = a.similarity / a.len; + const b_score = b.similarity / b.len; + if (a_score > b_score) + return -1; + if (a_score < b_score) + return 1; + return 0; + }); + return nearest; + } + get_nearest_until_next_dev_exceeds_std_dev(nearest) { + const sim = nearest.map((n) => n.similarity); + const mean = sim.reduce((a, b) => a + b) / sim.length; + let std_dev = Math.sqrt(sim.map((x) => Math.pow(x - mean, 2)).reduce((a, b) => a + b) / sim.length); + let slice_i = 0; + while (slice_i < nearest.length) { + const next = nearest[slice_i + 1]; + if (next) { + const next_dev = Math.abs(next.similarity - nearest[slice_i].similarity); + if (next_dev > std_dev) { + if (slice_i < 3) + std_dev = std_dev * 1.5; + else + break; + } + } + slice_i++; + } + nearest = nearest.slice(0, slice_i + 1); + return nearest; + } + // this.test_get_nearest_until_next_dev_exceeds_std_dev(); + // // test get_nearest_until_next_dev_exceeds_std_dev + // test_get_nearest_until_next_dev_exceeds_std_dev() { + // const nearest = [{similarity: 0.99}, {similarity: 0.98}, {similarity: 0.97}, {similarity: 0.96}, {similarity: 0.95}, {similarity: 0.94}, {similarity: 0.93}, {similarity: 0.92}, {similarity: 0.91}, {similarity: 0.9}, {similarity: 0.79}, {similarity: 0.78}, {similarity: 0.77}, {similarity: 0.76}, {similarity: 0.75}, {similarity: 0.74}, {similarity: 0.73}, {similarity: 0.72}]; + // const result = this.get_nearest_until_next_dev_exceeds_std_dev(nearest); + // if(result.length !== 10){ + // console.error("get_nearest_until_next_dev_exceeds_std_dev failed", result); + // } + // } + async get_context_for_prompt(nearest) { + let context = []; + const MAX_SOURCES = this.plugin.settings.smart_chat_model === "gpt-4-1106-preview" ? 42 : 20; + const MAX_CHARS = get_max_chars(this.plugin.settings.smart_chat_model) / 2; + let char_accum = 0; + for (let i = 0; i < nearest.length; i++) { + if (context.length >= MAX_SOURCES) + break; + if (char_accum >= MAX_CHARS) + break; + if (typeof nearest[i].link !== "string") + continue; + const breadcrumbs = nearest[i].link.replace(/#/g, " > ").replace(".md", "").replace(/\//g, " > "); + let new_context = `${breadcrumbs}: +`; + const max_available_chars = MAX_CHARS - char_accum - new_context.length; + if (nearest[i].link.indexOf("#") !== -1) { + new_context += await this.plugin.block_retriever(nearest[i].link, { max_chars: max_available_chars }); + } else { + new_context += await this.plugin.file_retriever(nearest[i].link, { max_chars: max_available_chars }); + } + char_accum += new_context.length; + context.push({ + link: nearest[i].link, + text: new_context + }); + } + console.log("context sources: " + context.length); + console.log("total context tokens: ~" + Math.round(char_accum / 3.5)); + this.chat.context = `Anticipate the type of answer desired by the user. Imagine the following ${context.length} notes were written by the user and contain all the necessary information to answer the user's question. Begin responses with "${SMART_TRANSLATION[this.plugin.settings.language].prompt}..."`; + for (let i = 0; i < context.length; i++) { + this.chat.context += ` +---BEGIN #${i + 1}--- +${context[i].text} +---END #${i + 1}---`; + } + return this.chat.context; + } +}; +function get_max_chars(model = "gpt-3.5-turbo") { + const MAX_CHAR_MAP = { + "gpt-3.5-turbo-16k": 48e3, + "gpt-4": 24e3, + "gpt-3.5-turbo": 12e3, + "gpt-4-1106-preview": 2e5 + }; + return MAX_CHAR_MAP[model]; +} +var SmartConnectionsChatModel = class { + constructor(plugin) { + this.app = plugin.app; + this.plugin = plugin; + this.chat_id = null; + this.chat_ml = []; + this.context = null; + this.hyd = null; + this.thread = []; + } + async save_chat() { + if (this.thread.length === 0) + return; + if (!await this.app.vault.adapter.exists(".smart-connections/chats")) { + await this.app.vault.adapter.mkdir(".smart-connections/chats"); + } + if (!this.chat_id) { + this.chat_id = this.name() + "\u2014" + this.get_file_date_string(); + } + if (!this.chat_id.match(/^[a-zA-Z0-9_—\- ]+$/)) { + console.log("Invalid chat_id: " + this.chat_id); + new Obsidian.Notice("[Smart Connections] Failed to save chat. Invalid chat_id: '" + this.chat_id + "'"); + } + const chat_file = this.chat_id + ".json"; + this.app.vault.adapter.write( + ".smart-connections/chats/" + chat_file, + JSON.stringify(this.thread, null, 2) + ); + } + async load_chat(chat_id) { + this.chat_id = chat_id; + const chat_file = this.chat_id + ".json"; + let chat_json = await this.app.vault.adapter.read( + ".smart-connections/chats/" + chat_file + ); + this.thread = JSON.parse(chat_json); + this.chat_ml = this.prepare_chat_ml(); + } + // prepare chat_ml from chat + // gets the last message of each turn unless turn_variation_offsets=[[turn_index,variation_index]] is specified in offset + prepare_chat_ml(turn_variation_offsets = []) { + if (turn_variation_offsets.length === 0) { + this.chat_ml = this.thread.map((turn) => { + return turn[turn.length - 1]; + }); + } else { + let turn_variation_index = []; + for (let i = 0; i < turn_variation_offsets.length; i++) { + turn_variation_index[turn_variation_offsets[i][0]] = turn_variation_offsets[i][1]; + } + this.chat_ml = this.thread.map((turn, turn_index) => { + if (turn_variation_index[turn_index] !== void 0) { + return turn[turn_variation_index[turn_index]]; + } + return turn[turn.length - 1]; + }); + } + this.chat_ml = this.chat_ml.map((message) => { + return { + role: message.role, + content: message.content + }; + }); + return this.chat_ml; + } + last() { + return this.thread[this.thread.length - 1][this.thread[this.thread.length - 1].length - 1]; + } + last_from() { + return this.last().role; + } + // returns user_input or completion + last_message() { + return this.last().content; + } + // message={} + // add new message to thread + new_message_in_thread(message, turn = -1) { + if (this.context) { + message.context = this.context; + this.context = null; + } + if (this.hyd) { + message.hyd = this.hyd; + this.hyd = null; + } + if (turn === -1) { + this.thread.push([message]); + } else { + this.thread[turn].push(message); + } + } + reset_context() { + this.context = null; + this.hyd = null; + } + async rename_chat(new_name) { + if (this.chat_id && await this.app.vault.adapter.exists(".smart-connections/chats/" + this.chat_id + ".json")) { + new_name = this.chat_id.replace(this.name(), new_name); + await this.app.vault.adapter.rename( + ".smart-connections/chats/" + this.chat_id + ".json", + ".smart-connections/chats/" + new_name + ".json" + ); + this.chat_id = new_name; + } else { + this.chat_id = new_name + "\u2014" + this.get_file_date_string(); + await this.save_chat(); + } + } + name() { + if (this.chat_id) { + return this.chat_id.replace(/—[^—]*$/, ""); + } + return "UNTITLED"; + } + get_file_date_string() { + return (/* @__PURE__ */ new Date()).toISOString().replace(/(T|:|\..*)/g, " ").trim(); + } + // get response from with note context + async get_response_with_note_context(user_input, chat_view) { + let system_input = "Imagine the following notes were written by the user and contain the necessary information to synthesize a useful answer the user's query:\n"; + const notes = this.extract_internal_links(user_input); + let max_chars = get_max_chars(this.plugin.settings.smart_chat_model); + for (let i = 0; i < notes.length; i++) { + const this_max_chars = notes.length - i > 1 ? Math.floor(max_chars / (notes.length - i)) : max_chars; + const note_content = await this.get_note_contents(notes[i], { char_limit: this_max_chars }); + console.log(note_content); + system_input += `---BEGIN NOTE: [[${notes[i].basename}]]--- +`; + system_input += note_content; + system_input += `---END NOTE--- +`; + max_chars -= note_content.length; + if (max_chars <= 0) + break; + } + this.context = system_input; + const chatml = [ + { + role: "system", + content: system_input + }, + { + role: "user", + content: user_input + } + ]; + chat_view.request_chatgpt_completion({ messages: chatml, temperature: 0 }); + } + // check if contains internal link + contains_internal_link(user_input) { + if (user_input.indexOf("[[") === -1) + return false; + if (user_input.indexOf("]]") === -1) + return false; + return true; + } + // check if contains folder reference (ex. /folder/, or /folder/subfolder/) + contains_folder_reference(user_input) { + if (user_input.indexOf("/") === -1) + return false; + if (user_input.indexOf("/") === user_input.lastIndexOf("/")) + return false; + return true; + } + // get folder references from user input + get_folder_references(user_input) { + const folders = this.plugin.folders.slice(); + const matches = folders.sort((a, b) => b.length - a.length).map((folder) => { + if (user_input.indexOf(folder) !== -1) { + user_input = user_input.replace(folder, ""); + return folder; + } + return false; + }).filter((folder) => folder); + console.log(matches); + if (matches) + return matches; + return false; + } + // extract internal links + extract_internal_links(user_input) { + const matches = user_input.match(/\[\[(.*?)\]\]/g); + console.log(matches); + if (matches) + return matches.map((match) => { + return this.app.metadataCache.getFirstLinkpathDest(match.replace("[[", "").replace("]]", ""), "/"); + }); + return []; + } + // get context from internal links + async get_note_contents(note, opts = {}) { + opts = { + char_limit: 1e4, + ...opts + }; + if (!(note instanceof Obsidian.TFile)) + return ""; + let file_content = await this.app.vault.cachedRead(note); + if (this.plugin.settings.cut_off_frontmatter) { + file_content = file_content.replace(/\s*---[\s\S]*?---/, ""); + } + if (file_content.indexOf("```dataview") > -1) { + file_content = await this.render_dataview_queries(file_content, note.path, opts); + } + return file_content.substring(0, opts.char_limit); + } + async render_dataview_queries(file_content, note_path, opts = {}) { + opts = { + char_limit: null, + ...opts + }; + const dataview_api = window["DataviewAPI"]; + if (!dataview_api) + return file_content; + const dataview_code_blocks = file_content.match(/```dataview(.*?)```/gs); + for (let i = 0; i < dataview_code_blocks.length; i++) { + if (opts.char_limit && opts.char_limit < file_content.indexOf(dataview_code_blocks[i])) + break; + const dataview_code_block = dataview_code_blocks[i]; + const dataview_code_block_content = dataview_code_block.replace("```dataview", "").replace("```", ""); + const dataview_query_result = await dataview_api.queryMarkdown(dataview_code_block_content, note_path, null); + if (dataview_query_result.successful) { + file_content = file_content.replace(dataview_code_block, dataview_query_result.value); + } + } + return file_content; + } +}; +var SmartConnectionsChatHistoryModal = class extends Obsidian.FuzzySuggestModal { + constructor(app, view, files) { + super(app); + this.app = app; + this.view = view; + this.setPlaceholder("Type the name of a chat session..."); + } + getItems() { + if (!this.view.files) { + return []; + } + return this.view.files; + } + getItemText(item) { + if (item.indexOf("UNTITLED") === -1) { + item.replace(/—[^—]*$/, ""); + } + return item; + } + onChooseItem(session) { + this.view.open_chat(session); + } +}; +var SmartConnectionsFileSelectModal = class extends Obsidian.FuzzySuggestModal { + constructor(app, view) { + super(app); + this.app = app; + this.view = view; + this.setPlaceholder("Type the name of a file..."); + } + getItems() { + return this.app.vault.getMarkdownFiles().sort((a, b) => a.basename.localeCompare(b.basename)); + } + getItemText(item) { + return item.basename; + } + onChooseItem(file) { + this.view.insert_selection(file.basename + "]] "); + } +}; +var SmartConnectionsFolderSelectModal = class extends Obsidian.FuzzySuggestModal { + constructor(app, view) { + super(app); + this.app = app; + this.view = view; + this.setPlaceholder("Type the name of a folder..."); + } + getItems() { + return this.view.plugin.folders; + } + getItemText(item) { + return item; + } + onChooseItem(folder) { + this.view.insert_selection(folder + "/ "); + } +}; +var ScStreamer = class { + // constructor + constructor(url, options) { + options = options || {}; + this.url = url; + this.method = options.method || "GET"; + this.headers = options.headers || {}; + this.payload = options.payload || null; + this.withCredentials = options.withCredentials || false; + this.listeners = {}; + this.readyState = this.CONNECTING; + this.progress = 0; + this.chunk = ""; + this.xhr = null; + this.FIELD_SEPARATOR = ":"; + this.INITIALIZING = -1; + this.CONNECTING = 0; + this.OPEN = 1; + this.CLOSED = 2; + } + // addEventListener + addEventListener(type, listener) { + if (!this.listeners[type]) { + this.listeners[type] = []; + } + if (this.listeners[type].indexOf(listener) === -1) { + this.listeners[type].push(listener); + } + } + // removeEventListener + removeEventListener(type, listener) { + if (!this.listeners[type]) { + return; + } + let filtered = []; + for (let i = 0; i < this.listeners[type].length; i++) { + if (this.listeners[type][i] !== listener) { + filtered.push(this.listeners[type][i]); + } + } + if (this.listeners[type].length === 0) { + delete this.listeners[type]; + } else { + this.listeners[type] = filtered; + } + } + // dispatchEvent + dispatchEvent(event) { + if (!event) { + return true; + } + event.source = this; + let onHandler = "on" + event.type; + if (this.hasOwnProperty(onHandler)) { + this[onHandler].call(this, event); + if (event.defaultPrevented) { + return false; + } + } + if (this.listeners[event.type]) { + return this.listeners[event.type].every(function(callback) { + callback(event); + return !event.defaultPrevented; + }); + } + return true; + } + // _setReadyState + _setReadyState(state) { + let event = new CustomEvent("readyStateChange"); + event.readyState = state; + this.readyState = state; + this.dispatchEvent(event); + } + // _onStreamFailure + _onStreamFailure(e) { + let event = new CustomEvent("error"); + event.data = e.currentTarget.response; + this.dispatchEvent(event); + this.close(); + } + // _onStreamAbort + _onStreamAbort(e) { + let event = new CustomEvent("abort"); + this.close(); + } + // _onStreamProgress + _onStreamProgress(e) { + if (!this.xhr) { + return; + } + if (this.xhr.status !== 200) { + this._onStreamFailure(e); + return; + } + if (this.readyState === this.CONNECTING) { + this.dispatchEvent(new CustomEvent("open")); + this._setReadyState(this.OPEN); + } + let data = this.xhr.responseText.substring(this.progress); + this.progress += data.length; + data.split(/(\r\n|\r|\n){2}/g).forEach(function(part) { + if (part.trim().length === 0) { + this.dispatchEvent(this._parseEventChunk(this.chunk.trim())); + this.chunk = ""; + } else { + this.chunk += part; + } + }.bind(this)); + } + // _onStreamLoaded + _onStreamLoaded(e) { + this._onStreamProgress(e); + this.dispatchEvent(this._parseEventChunk(this.chunk)); + this.chunk = ""; + } + // _parseEventChunk + _parseEventChunk(chunk) { + if (!chunk || chunk.length === 0) { + return null; + } + let e = { id: null, retry: null, data: "", event: "message" }; + chunk.split(/(\r\n|\r|\n)/).forEach(function(line) { + line = line.trimRight(); + let index = line.indexOf(this.FIELD_SEPARATOR); + if (index <= 0) { + return; + } + let field = line.substring(0, index); + if (!(field in e)) { + return; + } + let value = line.substring(index + 1).trimLeft(); + if (field === "data") { + e[field] += value; + } else { + e[field] = value; + } + }.bind(this)); + let event = new CustomEvent(e.event); + event.data = e.data; + event.id = e.id; + return event; + } + // _checkStreamClosed + _checkStreamClosed() { + if (!this.xhr) { + return; + } + if (this.xhr.readyState === XMLHttpRequest.DONE) { + this._setReadyState(this.CLOSED); + } + } + // stream + stream() { + this._setReadyState(this.CONNECTING); + this.xhr = new XMLHttpRequest(); + this.xhr.addEventListener("progress", this._onStreamProgress.bind(this)); + this.xhr.addEventListener("load", this._onStreamLoaded.bind(this)); + this.xhr.addEventListener("readystatechange", this._checkStreamClosed.bind(this)); + this.xhr.addEventListener("error", this._onStreamFailure.bind(this)); + this.xhr.addEventListener("abort", this._onStreamAbort.bind(this)); + this.xhr.open(this.method, this.url); + for (let header in this.headers) { + this.xhr.setRequestHeader(header, this.headers[header]); + } + this.xhr.withCredentials = this.withCredentials; + this.xhr.send(this.payload); + } + // close + close() { + if (this.readyState === this.CLOSED) { + return; + } + this.xhr.abort(); + this.xhr = null; + this._setReadyState(this.CLOSED); + } +}; +module.exports = SmartConnectionsPlugin; +//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/index.js"],
  "sourcesContent": ["const Obsidian = require(\"obsidian\");\r\nconst VecLite = class {\r\n  constructor(config) {\r\n    this.config = {\r\n      file_name: \"embeddings-3.json\",\r\n      folder_path: \".vec_lite\",\r\n      exists_adapter: null,\r\n      mkdir_adapter: null,\r\n      read_adapter: null,\r\n      rename_adapter: null,\r\n      stat_adapter: null,\r\n      write_adapter: null,\r\n      ...config,\r\n    };\r\n    this.file_name = this.config.file_name;\r\n    this.folder_path = config.folder_path;\r\n    this.file_path = this.folder_path + \"/\" + this.file_name;\r\n    this.embeddings = false;\r\n  }\r\n  async file_exists(path) {\r\n    if (this.config.exists_adapter) {\r\n      return await this.config.exists_adapter(path);\r\n    } else {\r\n      throw new Error(\"exists_adapter not set\");\r\n    }\r\n  }\r\n  async mkdir(path) {\r\n    if (this.config.mkdir_adapter) {\r\n      return await this.config.mkdir_adapter(path);\r\n    } else {\r\n      throw new Error(\"mkdir_adapter not set\");\r\n    }\r\n  }\r\n  async read_file(path) {\r\n    if (this.config.read_adapter) {\r\n      return await this.config.read_adapter(path);\r\n    } else {\r\n      throw new Error(\"read_adapter not set\");\r\n    }\r\n  }\r\n  async rename(old_path, new_path) {\r\n    if (this.config.rename_adapter) {\r\n      return await this.config.rename_adapter(old_path, new_path);\r\n    } else {\r\n      throw new Error(\"rename_adapter not set\");\r\n    }\r\n  }\r\n  async stat(path) {\r\n    if (this.config.stat_adapter) {\r\n      return await this.config.stat_adapter(path);\r\n    } else {\r\n      throw new Error(\"stat_adapter not set\");\r\n    }\r\n  }\r\n  async write_file(path, data) {\r\n    if (this.config.write_adapter) {\r\n      return await this.config.write_adapter(path, data);\r\n    } else {\r\n      throw new Error(\"write_adapter not set\");\r\n    }\r\n  }\r\n  async load(retries = 0) {\r\n    try {\r\n      const embeddings_file = await this.read_file(this.file_path);\r\n      this.embeddings = JSON.parse(embeddings_file);\r\n      console.log(\"loaded embeddings file: \" + this.file_path);\r\n      return true;\r\n    } catch (error) {\r\n      if (retries < 3) {\r\n        console.log(\"retrying load()\");\r\n        await new Promise((r) => setTimeout(r, 1e3 + 1e3 * retries));\r\n        return await this.load(retries + 1);\r\n      }\r\n      console.log(\r\n        \"failed to load embeddings file, prompt user to initiate bulk embed\"\r\n      );\r\n      return false;\r\n    }\r\n  }\r\n  async init_embeddings_file() {\r\n    if (!(await this.file_exists(this.folder_path))) {\r\n      await this.mkdir(this.folder_path);\r\n      console.log(\"created folder: \" + this.folder_path);\r\n    } else {\r\n      console.log(\"folder already exists: \" + this.folder_path);\r\n    }\r\n    if (!(await this.file_exists(this.file_path))) {\r\n      await this.write_file(this.file_path, \"{}\");\r\n      console.log(\"created embeddings file: \" + this.file_path);\r\n    } else {\r\n      console.log(\"embeddings file already exists: \" + this.file_path);\r\n    }\r\n  }\r\n  async save() {\r\n    const embeddings = JSON.stringify(this.embeddings);\r\n    const embeddings_file_exists = await this.file_exists(this.file_path);\r\n    if (embeddings_file_exists) {\r\n      const new_file_size = embeddings.length;\r\n      const existing_file_size = await this.stat(this.file_path).then(\r\n        (stat) => stat.size\r\n      );\r\n      if (new_file_size > existing_file_size * 0.5) {\r\n        await this.write_file(this.file_path, embeddings);\r\n        console.log(\"embeddings file size: \" + new_file_size + \" bytes\");\r\n      } else {\r\n        const warning_message = [\r\n          \"Warning: New embeddings file size is significantly smaller than existing embeddings file size.\",\r\n          \"Aborting to prevent possible loss of embeddings data.\",\r\n          \"New file size: \" + new_file_size + \" bytes.\",\r\n          \"Existing file size: \" + existing_file_size + \" bytes.\",\r\n          \"Restarting Obsidian may fix this.\",\r\n        ];\r\n        console.log(warning_message.join(\" \"));\r\n        await this.write_file(\r\n          this.folder_path + \"/unsaved-embeddings.json\",\r\n          embeddings\r\n        );\r\n        throw new Error(\r\n          \"Error: New embeddings file size is significantly smaller than existing embeddings file size. Aborting to prevent possible loss of embeddings data.\"\r\n        );\r\n      }\r\n    } else {\r\n      await this.init_embeddings_file();\r\n      return await this.save();\r\n    }\r\n    return true;\r\n  }\r\n  cos_sim(vector1, vector2) {\r\n    let dotProduct = 0;\r\n    let normA = 0;\r\n    let normB = 0;\r\n    for (let i = 0; i < vector1.length; i++) {\r\n      dotProduct += vector1[i] * vector2[i];\r\n      normA += vector1[i] * vector1[i];\r\n      normB += vector2[i] * vector2[i];\r\n    }\r\n    if (normA === 0 || normB === 0) {\r\n      return 0;\r\n    } else {\r\n      return dotProduct / (Math.sqrt(normA) * Math.sqrt(normB));\r\n    }\r\n  }\r\n  nearest(to_vec, filter = {}) {\r\n    filter = {\r\n      results_count: 30,\r\n      ...filter,\r\n    };\r\n    let nearest = [];\r\n    const from_keys = Object.keys(this.embeddings);\r\n    for (let i = 0; i < from_keys.length; i++) {\r\n      if (filter.skip_sections) {\r\n        const from_path = this.embeddings[from_keys[i]].meta.path;\r\n        if (from_path.indexOf(\"#\") > -1) continue;\r\n      }\r\n      if (filter.skip_key) {\r\n        if (filter.skip_key === from_keys[i]) continue;\r\n        if (filter.skip_key === this.embeddings[from_keys[i]].meta.parent)\r\n          continue;\r\n      }\r\n      if (filter.path_begins_with) {\r\n        if (\r\n          typeof filter.path_begins_with === \"string\" &&\r\n          !this.embeddings[from_keys[i]].meta.path.startsWith(\r\n            filter.path_begins_with\r\n          )\r\n        )\r\n          continue;\r\n        if (\r\n          Array.isArray(filter.path_begins_with) &&\r\n          !filter.path_begins_with.some((path) =>\r\n            this.embeddings[from_keys[i]].meta.path.startsWith(path)\r\n          )\r\n        )\r\n          continue;\r\n      }\r\n      nearest.push({\r\n        link: this.embeddings[from_keys[i]].meta.path,\r\n        similarity: this.cos_sim(to_vec, this.embeddings[from_keys[i]].vec),\r\n        size: this.embeddings[from_keys[i]].meta.size,\r\n      });\r\n    }\r\n    nearest.sort(function (a, b) {\r\n      return b.similarity - a.similarity;\r\n    });\r\n    nearest = nearest.slice(0, filter.results_count);\r\n    return nearest;\r\n  }\r\n  find_nearest_embeddings(to_vec, filter = {}) {\r\n    const default_filter = {\r\n      max: this.max_sources,\r\n    };\r\n    filter = { ...default_filter, ...filter };\r\n    if (Array.isArray(to_vec) && to_vec.length !== this.vec_len) {\r\n      this.nearest = {};\r\n      for (let i = 0; i < to_vec.length; i++) {\r\n        this.find_nearest_embeddings(to_vec[i], {\r\n          max: Math.floor(filter.max / to_vec.length),\r\n        });\r\n      }\r\n    } else {\r\n      const from_keys = Object.keys(this.embeddings);\r\n      for (let i = 0; i < from_keys.length; i++) {\r\n        if (this.validate_type(this.embeddings[from_keys[i]])) continue;\r\n        const sim = this.computeCosineSimilarity(\r\n          to_vec,\r\n          this.embeddings[from_keys[i]].vec\r\n        );\r\n        if (this.nearest[from_keys[i]]) {\r\n          this.nearest[from_keys[i]] += sim;\r\n        } else {\r\n          this.nearest[from_keys[i]] = sim;\r\n        }\r\n      }\r\n    }\r\n    let nearest = Object.keys(this.nearest).map((key) => {\r\n      return {\r\n        key,\r\n        similarity: this.nearest[key],\r\n      };\r\n    });\r\n    nearest = this.sort_by_similarity(nearest);\r\n    nearest = nearest.slice(0, filter.max);\r\n    nearest = nearest.map((item) => {\r\n      return {\r\n        link: this.embeddings[item.key].meta.path,\r\n        similarity: item.similarity,\r\n        len:\r\n          this.embeddings[item.key].meta.len ||\r\n          this.embeddings[item.key].meta.size,\r\n      };\r\n    });\r\n    return nearest;\r\n  }\r\n  sort_by_similarity(nearest) {\r\n    return nearest.sort(function (a, b) {\r\n      const a_score = a.similarity;\r\n      const b_score = b.similarity;\r\n      if (a_score > b_score) return -1;\r\n      if (a_score < b_score) return 1;\r\n      return 0;\r\n    });\r\n  }\r\n  // check if key from embeddings exists in files\r\n  clean_up_embeddings(files) {\r\n    console.log(\"cleaning up embeddings\");\r\n    const keys = Object.keys(this.embeddings);\r\n    let deleted_embeddings = 0;\r\n    for (const key of keys) {\r\n      const path = this.embeddings[key].meta.path;\r\n      if (!files.find((file) => path.startsWith(file.path))) {\r\n        delete this.embeddings[key];\r\n        deleted_embeddings++;\r\n        continue;\r\n      }\r\n      if (path.indexOf(\"#\") > -1) {\r\n        const parent_key = this.embeddings[key].meta.parent;\r\n        if (!this.embeddings[parent_key]) {\r\n          delete this.embeddings[key];\r\n          deleted_embeddings++;\r\n          continue;\r\n        }\r\n        if (!this.embeddings[parent_key].meta) {\r\n          delete this.embeddings[key];\r\n          deleted_embeddings++;\r\n          continue;\r\n        }\r\n        if (\r\n          this.embeddings[parent_key].meta.children &&\r\n          this.embeddings[parent_key].meta.children.indexOf(key) < 0\r\n        ) {\r\n          delete this.embeddings[key];\r\n          deleted_embeddings++;\r\n          continue;\r\n        }\r\n      }\r\n    }\r\n    return { deleted_embeddings, total_embeddings: keys.length };\r\n  }\r\n  get(key) {\r\n    return this.embeddings[key] || null;\r\n  }\r\n  get_meta(key) {\r\n    const embedding = this.get(key);\r\n    if (embedding && embedding.meta) {\r\n      return embedding.meta;\r\n    }\r\n    return null;\r\n  }\r\n  get_mtime(key) {\r\n    const meta = this.get_meta(key);\r\n    if (meta && meta.mtime) {\r\n      return meta.mtime;\r\n    }\r\n    return null;\r\n  }\r\n  get_hash(key) {\r\n    const meta = this.get_meta(key);\r\n    if (meta && meta.hash) {\r\n      return meta.hash;\r\n    }\r\n    return null;\r\n  }\r\n  get_size(key) {\r\n    const meta = this.get_meta(key);\r\n    if (meta && meta.size) {\r\n      return meta.size;\r\n    }\r\n    return null;\r\n  }\r\n  get_children(key) {\r\n    const meta = this.get_meta(key);\r\n    if (meta && meta.children) {\r\n      return meta.children;\r\n    }\r\n    return null;\r\n  }\r\n  get_vec(key) {\r\n    const embedding = this.get(key);\r\n    if (embedding && embedding.vec) {\r\n      return embedding.vec;\r\n    }\r\n    return null;\r\n  }\r\n  save_embedding(key, vec, meta) {\r\n    this.embeddings[key] = {\r\n      vec,\r\n      meta,\r\n    };\r\n  }\r\n  mtime_is_current(key, source_mtime) {\r\n    const mtime = this.get_mtime(key);\r\n    if (mtime && mtime >= source_mtime) {\r\n      return true;\r\n    }\r\n    return false;\r\n  }\r\n  async force_refresh() {\r\n    this.embeddings = null;\r\n    this.embeddings = {};\r\n    let current_datetime = Math.floor(Date.now() / 1e3);\r\n    await this.rename(\r\n      this.file_path,\r\n      this.folder_path + \"/embeddings-\" + current_datetime + \".json\"\r\n    );\r\n    await this.init_embeddings_file();\r\n  }\r\n};\r\n\r\n\r\n\r\nconst DEFAULT_SETTINGS = {\r\n  api_key: \"\",\r\n  chat_open: true,\r\n  file_exclusions: \"\",\r\n  folder_exclusions: \"\",\r\n  header_exclusions: \"\",\r\n  path_only: \"\",\r\n  show_full_path: false,\r\n  cut_off_frontmatter: false,\r\n  expanded_view: true,\r\n  group_nearest_by_file: false,\r\n  language: \"en\",\r\n  log_render: false,\r\n  log_render_files: false,\r\n  recently_sent_retry_notice: false,\r\n  skip_sections: false,\r\n  smart_chat_model: \"gpt-3.5-turbo-16k\",\r\n  view_open: true,\r\n  version: \"\",\r\n  open_in_big_view: false,\r\n};\r\nconst MAX_EMBED_STRING_LENGTH = 25000;\r\n\r\nlet VERSION;\r\nconst SUPPORTED_FILE_TYPES = [\"md\", \"canvas\"];\r\n\r\n//create one object with all the translations\r\n// research : SMART_TRANSLATION[language][key]\r\nconst SMART_TRANSLATION = {\r\n  \"en\": {\r\n    \"pronous\": [\"my\", \"I\", \"me\", \"mine\", \"our\", \"ours\", \"us\", \"we\"],\r\n    \"prompt\": \"Based on your notes\",\r\n    \"initial_message\": \"Hi, I'm ChatGPT with access to your notes via Smart Connections. Ask me a question about your notes and I'll try to answer it.\",\r\n    \"try_placeholder\": `Try \"Based on my notes\" or \"Summarize [[this note]]\" or \"Important tasks in /folder/\"`\r\n  },\r\n  \"es\": {\r\n    \"pronous\": [\"mi\", \"yo\", \"m\u00ED\", \"t\u00FA\"],\r\n    \"prompt\": \"Bas\u00E1ndose en sus notas\",\r\n    \"initial_message\": \"Hola, soy ChatGPT con acceso a tus apuntes a trav\u00E9s de Smart Connections. Hazme una pregunta sobre tus apuntes e intentar\u00E9 responderte.\",\r\n    \"try_placeholder\": `Prueba \"Basado en mis notas\" o \"Resumen [[esta nota]]\" o \"Tareas importantes en /carpeta/\"`\r\n  },\r\n  \"fr\": {\r\n    \"pronous\": [\"me\", \"mon\", \"ma\", \"mes\", \"moi\", \"nous\", \"notre\", \"nos\", \"je\", \"j'\", \"m'\"],\r\n    \"prompt\": \"D'apr\u00E8s vos notes\",\r\n    \"initial_message\": \"Bonjour, je suis ChatGPT et j'ai acc\u00E8s \u00E0 vos notes via Smart Connections. Posez-moi une question sur vos notes et j'essaierai d'y r\u00E9pondre.\",\r\n    \"try_placeholder\": `Essayez \"D'apr\u00E8s mes notes\" ou \"R\u00E9sume [[cette note]]\" ou \"T\u00E2ches importantes dans /dossier/\"`\r\n  },\r\n  \"de\": {\r\n    \"pronous\": [\"mein\", \"meine\", \"meinen\", \"meiner\", \"meines\", \"mir\", \"uns\", \"unser\", \"unseren\", \"unserer\", \"unseres\"],\r\n    \"prompt\": \"Basierend auf Ihren Notizen\",\r\n    \"initial_message\": \"Hallo, ich bin ChatGPT und habe \u00FCber Smart Connections Zugang zu Ihren Notizen. Stellen Sie mir eine Frage zu Ihren Notizen und ich werde versuchen, sie zu beantworten.\",\r\n    \"try_placeholder\": `Versuchen Sie \"Basierend auf meinen Notizen\" oder \"Zusammenfassen [[dieser Notiz]]\" oder \"Wichtige Aufgaben im /Ordner/\"`\r\n  },\r\n  \"it\": {\r\n    \"pronous\": [\"mio\", \"mia\", \"miei\", \"mie\", \"noi\", \"nostro\", \"nostri\", \"nostra\", \"nostre\"],\r\n    \"prompt\": \"Sulla base degli appunti\",\r\n    \"initial_message\": \"Ciao, sono ChatGPT e ho accesso ai tuoi appunti tramite Smart Connections. Fatemi una domanda sui vostri appunti e cercher\u00F2 di rispondervi.\",\r\n    \"try_placeholder\": `Prova \"Sulla base dei miei appunti\" o \"Riassumi [[questo appunto]]\" o \"Compiti importanti in /cartella/\"`\r\n  },\r\n}\r\n\r\n// require built-in crypto module\r\nconst crypto = require(\"crypto\");\r\n// md5 hash using built in crypto module\r\nfunction md5(str) {\r\n  return crypto.createHash(\"md5\").update(str).digest(\"hex\");\r\n}\r\n\r\nclass SmartConnectionsPlugin extends Obsidian.Plugin {\r\n  // constructor\r\n  constructor() {\r\n    super(...arguments);\r\n    this.api = null;\r\n    this.embeddings_loaded = false;\r\n    this.file_exclusions = [];\r\n    this.folders = [];\r\n    this.has_new_embeddings = false;\r\n    this.header_exclusions = [];\r\n    this.nearest_cache = {};\r\n    this.path_only = [];\r\n    this.render_log = {};\r\n    this.render_log.deleted_embeddings = 0;\r\n    this.render_log.exclusions_logs = {};\r\n    this.render_log.failed_embeddings = [];\r\n    this.render_log.files = [];\r\n    this.render_log.new_embeddings = 0;\r\n    this.render_log.skipped_low_delta = {};\r\n    this.render_log.token_usage = 0;\r\n    this.render_log.tokens_saved_by_cache = 0;\r\n    this.retry_notice_timeout = null;\r\n    this.save_timeout = null;\r\n    this.sc_branding = {};\r\n    this.self_ref_kw_regex = null;\r\n    this.update_available = false;\r\n  }\r\n\r\n  async onload() {\r\n    // initialize when layout is ready\r\n    this.app.workspace.onLayoutReady(this.initialize.bind(this));\r\n  }\r\n  onunload() {\r\n    this.output_render_log();\r\n    console.log(\"unloading plugin\");\r\n    this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE);\r\n    this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE);\r\n  }\r\n  async initialize() {\r\n    console.log(\"Loading Smart Connections plugin\");\r\n    VERSION = this.manifest.version;\r\n    // VERSION = '1.0.0';\r\n    // console.log(VERSION);\r\n    await this.loadSettings();\r\n    // run after 3 seconds\r\n    setTimeout(this.check_for_update.bind(this), 3000);\r\n    // run check for update every 3 hours\r\n    setInterval(this.check_for_update.bind(this), 10800000);\r\n\r\n    this.addIcon();\r\n    this.addCommand({\r\n      id: \"sc-find-notes\",\r\n      name: \"Find: Make Smart Connections\",\r\n      icon: \"pencil_icon\",\r\n      hotkeys: [],\r\n      // editorCallback: async (editor) => {\r\n      editorCallback: async (editor) => {\r\n        if(editor.somethingSelected()) {\r\n          // get selected text\r\n          let selected_text = editor.getSelection();\r\n          // render connections from selected text\r\n          await this.make_connections(selected_text);\r\n        } else {\r\n          // clear nearest_cache on manual call to make connections\r\n          this.nearest_cache = {};\r\n          // console.log(\"Cleared nearest_cache\");\r\n          await this.make_connections();\r\n        }\r\n      }\r\n    });\r\n    this.addCommand({\r\n      id: \"smart-connections-view\",\r\n      name: \"Open: View Smart Connections\",\r\n      callback: () => {\r\n        this.open_view();\r\n      }\r\n    });\r\n    // open chat command\r\n    this.addCommand({\r\n      id: \"smart-connections-chat\",\r\n      name: \"Open: Smart Chat Conversation\",\r\n      callback: () => {\r\n        this.open_chat();\r\n      }\r\n    });\r\n    // open random note from nearest cache\r\n    this.addCommand({\r\n      id: \"smart-connections-random\",\r\n      name: \"Open: Random Note from Smart Connections\",\r\n      callback: () => {\r\n        this.open_random_note();\r\n      }\r\n    });\r\n    // add settings tab\r\n    this.addSettingTab(new SmartConnectionsSettingsTab(this.app, this));\r\n    // register main view type\r\n    this.registerView(SMART_CONNECTIONS_VIEW_TYPE, (leaf) => (new SmartConnectionsView(leaf, this)));\r\n    // register chat view type\r\n    this.registerView(SMART_CONNECTIONS_CHAT_VIEW_TYPE, (leaf) => (new SmartConnectionsChatView(leaf, this)));\r\n    // code-block renderer\r\n    this.registerMarkdownCodeBlockProcessor(\"smart-connections\", this.render_code_block.bind(this));\r\n\r\n    // if this settings.view_open is true, open view on startup\r\n    if(this.settings.view_open) {\r\n      this.open_view();\r\n    }\r\n    // if this settings.chat_open is true, open chat on startup\r\n    if(this.settings.chat_open) {\r\n      this.open_chat();\r\n    }\r\n    // on new version\r\n    if(this.settings.version !== VERSION) {\r\n      // update version\r\n      this.settings.version = VERSION;\r\n      // save settings\r\n      await this.saveSettings();\r\n      // open view\r\n      this.open_view();\r\n    }\r\n    // check github release endpoint if update is available\r\n    this.add_to_gitignore();\r\n    /**\r\n     * EXPERIMENTAL\r\n     * - window-based API access\r\n     * - code-block rendering\r\n     */\r\n    this.api = new ScSearchApi(this.app, this);\r\n    // register API to global window object\r\n    (window[\"SmartSearchApi\"] = this.api) && this.register(() => delete window[\"SmartSearchApi\"]);\r\n\r\n  }\r\n\r\n  async init_vecs() {\r\n    this.smart_vec_lite = new VecLite({\r\n      folder_path: \".smart-connections\",\r\n      exists_adapter: this.app.vault.adapter.exists.bind(this.app.vault.adapter),\r\n      mkdir_adapter: this.app.vault.adapter.mkdir.bind(this.app.vault.adapter),\r\n      read_adapter: this.app.vault.adapter.read.bind(this.app.vault.adapter),\r\n      rename_adapter: this.app.vault.adapter.rename.bind(this.app.vault.adapter),\r\n      stat_adapter: this.app.vault.adapter.stat.bind(this.app.vault.adapter),\r\n      write_adapter: this.app.vault.adapter.write.bind(this.app.vault.adapter),\r\n    });\r\n    this.embeddings_loaded = await this.smart_vec_lite.load();\r\n    return this.embeddings_loaded;\r\n  }\r\n  async update_to_v2() {\r\n    // if license key is not set, return\r\n    if(!this.settings.license_key) return new Obsidian.Notice(\"[Smart Connections] Supporter license key required for early access to V2\");\r\n    // download https://github.com/brianpetro/obsidian-smart-connections/releases/download/1.6.37/main.js\r\n    const v2 = await (0, Obsidian.requestUrl)({\r\n      url: \"https://sync.smartconnections.app/download_v2\",\r\n      method: \"POST\",\r\n      headers: {\r\n        \"Content-Type\": \"application/json\",\r\n      },\r\n      body: JSON.stringify({\r\n        license_key: this.settings.license_key,\r\n      })\r\n    });\r\n    if(v2.status !== 200) return console.error(\"Error downloading version 2\", v2);\r\n    console.log(v2);\r\n    await this.app.vault.adapter.write(\".obsidian/plugins/smart-connections/main.js\", v2.json.main); // add new\r\n    await this.app.vault.adapter.write(\".obsidian/plugins/smart-connections/manifest.json\", v2.json.manifest); // add new\r\n    await this.app.vault.adapter.write(\".obsidian/plugins/smart-connections/styles.css\", v2.json.styles); // add new\r\n    window.restart_plugin = async (id) => {\r\n      console.log(\"restarting plugin\", id);\r\n      await window.app.plugins.disablePlugin(id);\r\n      await window.app.plugins.enablePlugin(id);\r\n      console.log(\"plugin restarted\", id);\r\n    }\r\n    window.restart_plugin(this.manifest.id);\r\n  }\r\n\r\n  async loadSettings() {\r\n    this.settings = Object.assign({}, DEFAULT_SETTINGS, await this.loadData());\r\n    // load file exclusions if not blank\r\n    if(this.settings.file_exclusions && this.settings.file_exclusions.length > 0) {\r\n      // split file exclusions into array and trim whitespace\r\n      this.file_exclusions = this.settings.file_exclusions.split(/[,\\n\\s]/)\r\n        .filter((file) => file.length > 0)\r\n        .map((file) => {\r\n        return file.trim();\r\n      });\r\n    }\r\n    // load folder exclusions if not blank\r\n    if(this.settings.folder_exclusions && this.settings.folder_exclusions.length > 0) {\r\n      // add slash to end of folder name if not present\r\n      const folder_exclusions = this.settings.folder_exclusions\r\n        .split(/[,\\n\\s]/)\r\n        .filter((file) => file.length > 0)\r\n        .map((folder) => {\r\n          folder = folder.trim();\r\n          return folder.slice(-1) === \"/\" ? folder : `${folder}/`;\r\n        });\r\n      // merge folder exclusions with file exclusions\r\n      this.file_exclusions = this.file_exclusions.concat(folder_exclusions);\r\n    }\r\n    // load header exclusions if not blank\r\n    if(this.settings.header_exclusions && this.settings.header_exclusions.length > 0) {\r\n      this.header_exclusions = this.settings.header_exclusions\r\n        .split(/[\\s\\n,]/)\r\n        .filter((file) => file.length > 0)\r\n        .map((header) => {\r\n          return header.trim();\r\n        });\r\n    }\r\n    // load path_only if not blank\r\n    if(this.settings.path_only && this.settings.path_only.length > 0) {\r\n      this.path_only = this.settings.path_only\r\n        .split(/[\\s\\n,]/)\r\n        .filter((file) => file.length > 0)\r\n        .map((path) => {\r\n          return path.trim();\r\n        });\r\n    }\r\n    // load self_ref_kw_regex\r\n    this.self_ref_kw_regex = new RegExp(`\\\\b(${SMART_TRANSLATION[this.settings.language].pronous.join(\"|\")})\\\\b`, \"gi\");\r\n    // load failed files\r\n    await this.load_failed_files();\r\n  }\r\n  async saveSettings(rerender=false) {\r\n    await this.saveData(this.settings);\r\n    // re-load settings into memory\r\n    await this.loadSettings();\r\n    // re-render view if set to true (for example, after adding API key)\r\n    if(rerender) {\r\n      this.nearest_cache = {};\r\n      await this.make_connections();\r\n    }\r\n  }\r\n\r\n  // check for update\r\n  async check_for_update() {\r\n    // fail silently, ex. if no internet connection\r\n    try {\r\n      // get latest release version from github\r\n      const response = await (0, Obsidian.requestUrl)({\r\n        url: \"https://api.github.com/repos/brianpetro/obsidian-smart-connections/releases/latest\",\r\n        method: \"GET\",\r\n        headers: {\r\n          \"Content-Type\": \"application/json\",\r\n        },\r\n        contentType: \"application/json\",\r\n      });\r\n      // get version number from response\r\n      const latest_release = JSON.parse(response.text).tag_name;\r\n      // console.log(`Latest release: ${latest_release}`);\r\n      // if latest_release is newer than current version, show message\r\n      if(latest_release !== VERSION) {\r\n        new Obsidian.Notice(`[Smart Connections] A new version is available! (v${latest_release})`);\r\n        this.update_available = true;\r\n        this.render_brand(\"all\")\r\n      }\r\n    } catch (error) {\r\n      console.log(error);\r\n    }\r\n  }\r\n\r\n  async render_code_block(contents, container, ctx) {\r\n    let nearest;\r\n    if(contents.trim().length > 0) {\r\n      nearest = await this.api.search(contents);\r\n    } else {\r\n      // use ctx to get file\r\n      console.log(ctx);\r\n      const file = this.app.vault.getAbstractFileByPath(ctx.sourcePath);\r\n      nearest = await this.find_note_connections(file);\r\n    }\r\n    if (nearest.length) {\r\n      this.update_results(container, nearest);\r\n    }\r\n  }\r\n\r\n  async make_connections(selected_text=null) {\r\n    let view = this.get_view();\r\n    if (!view) {\r\n      // open view if not open\r\n      await this.open_view();\r\n      view = this.get_view();\r\n    }\r\n    await view.render_connections(selected_text);\r\n  }\r\n\r\n  addIcon(){\r\n    Obsidian.addIcon(\"smart-connections\", `<path d=\"M50,20 L80,40 L80,60 L50,100\" stroke=\"currentColor\" stroke-width=\"4\" fill=\"none\"/>\r\n    <path d=\"M30,50 L55,70\" stroke=\"currentColor\" stroke-width=\"5\" fill=\"none\"/>\r\n    <circle cx=\"50\" cy=\"20\" r=\"9\" fill=\"currentColor\"/>\r\n    <circle cx=\"80\" cy=\"40\" r=\"9\" fill=\"currentColor\"/>\r\n    <circle cx=\"80\" cy=\"70\" r=\"9\" fill=\"currentColor\"/>\r\n    <circle cx=\"50\" cy=\"100\" r=\"9\" fill=\"currentColor\"/>\r\n    <circle cx=\"30\" cy=\"50\" r=\"9\" fill=\"currentColor\"/>`);\r\n  }\r\n\r\n  // open random note\r\n  async open_random_note() {\r\n    const curr_file = this.app.workspace.getActiveFile();\r\n    const curr_key = md5(curr_file.path);\r\n    // if no nearest cache, create Obsidian notice\r\n    if(typeof this.nearest_cache[curr_key] === \"undefined\") {\r\n      new Obsidian.Notice(\"[Smart Connections] No Smart Connections found. Open a note to get Smart Connections.\");\r\n      return;\r\n    }\r\n    // get random from nearest cache\r\n    const rand = Math.floor(Math.random() * this.nearest_cache[curr_key].length/2); // divide by 2 to limit to top half of results\r\n    const random_file = this.nearest_cache[curr_key][rand];\r\n    // open random file\r\n    this.open_note(random_file);\r\n  }\r\n\r\n  async open_view() {\r\n    if(this.get_view()){\r\n      console.log(\"Smart Connections view already open\");\r\n      return;\r\n    }\r\n    this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE);\r\n    await this.app.workspace.getRightLeaf(false).setViewState({\r\n      type: SMART_CONNECTIONS_VIEW_TYPE,\r\n      active: true,\r\n    });\r\n    this.app.workspace.revealLeaf(\r\n      this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE)[0]\r\n    );\r\n  }\r\n  // source: https://github.com/obsidianmd/obsidian-releases/blob/master/plugin-review.md#avoid-managing-references-to-custom-views\r\n  get_view() {\r\n    for (let leaf of this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE)) {\r\n      if (leaf.view instanceof SmartConnectionsView) {\r\n        return leaf.view;\r\n      }\r\n    }\r\n  }\r\n  // open chat view\r\n  async open_chat(retries=0) {\r\n    if(!this.embeddings_loaded) {\r\n      console.log(\"embeddings not loaded yet\");\r\n      if(retries < 3) {\r\n        // wait and try again\r\n        setTimeout(() => {\r\n          this.open_chat(retries+1);\r\n        }, 1000 * (retries+1));\r\n        return;\r\n      }\r\n      console.log(\"embeddings still not loaded, opening smart view\");\r\n      this.open_view();\r\n      return;\r\n    }\r\n    this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE);\r\n    if (!this.settings.open_in_big_view) {\r\n      await this.app.workspace.getRightLeaf(false).setViewState({\r\n        type: SMART_CONNECTIONS_CHAT_VIEW_TYPE,\r\n        active: true,\r\n      });\r\n    } else {\r\n      await this.app.workspace.getLeaf(true).setViewState({\r\n        type: SMART_CONNECTIONS_CHAT_VIEW_TYPE,\r\n        active: true,\r\n      })\r\n    }\r\n    this.app.workspace.revealLeaf(\r\n      this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE)[0]\r\n    );\r\n  }\r\n\r\n  // get embeddings for all files\r\n  async get_all_embeddings() {\r\n    // get all files in vault and filter all but markdown and canvas files\r\n    const files = (await this.app.vault.getFiles()).filter((file) => file instanceof Obsidian.TFile && (file.extension === \"md\" || file.extension === \"canvas\"));\r\n    // const files = await this.app.vault.getMarkdownFiles();\r\n    // get open files to skip if file is currently open\r\n    const open_files = this.app.workspace.getLeavesOfType(\"markdown\").map((leaf) => leaf.view.file);\r\n    const clean_up_log = this.smart_vec_lite.clean_up_embeddings(files);\r\n    if(this.settings.log_render){\r\n      this.render_log.total_files = files.length;\r\n      this.render_log.deleted_embeddings = clean_up_log.deleted_embeddings;\r\n      this.render_log.total_embeddings = clean_up_log.total_embeddings;\r\n    }\r\n    // batch embeddings\r\n    let batch_promises = [];\r\n    for (let i = 0; i < files.length; i++) {\r\n      // skip if path contains a #\r\n      if(files[i].path.indexOf(\"#\") > -1) {\r\n        // console.log(\"skipping file '\"+files[i].path+\"' (path contains #)\");\r\n        this.log_exclusion(\"path contains #\");\r\n        continue;\r\n      }\r\n      // skip if file already has embedding and embedding.mtime is greater than or equal to file.mtime\r\n      if(this.smart_vec_lite.mtime_is_current(md5(files[i].path), files[i].stat.mtime)) {\r\n        // log skipping file\r\n        // console.log(\"skipping file (mtime)\");\r\n        continue;\r\n      }\r\n      // check if file is in failed_files\r\n      if(this.settings.failed_files.indexOf(files[i].path) > -1) {\r\n        // log skipping file\r\n        // console.log(\"skipping previously failed file, use button in settings to retry\");\r\n        // use setTimeout to prevent multiple notices\r\n        if(this.retry_notice_timeout) {\r\n          clearTimeout(this.retry_notice_timeout);\r\n          this.retry_notice_timeout = null;\r\n        }\r\n        // limit to one notice every 10 minutes\r\n        if(!this.recently_sent_retry_notice){\r\n          new Obsidian.Notice(\"Smart Connections: Skipping previously failed file, use button in settings to retry\");\r\n          this.recently_sent_retry_notice = true;\r\n          setTimeout(() => {\r\n            this.recently_sent_retry_notice = false;\r\n          }, 600000);\r\n        }\r\n        continue;\r\n      }\r\n      // skip files where path contains any exclusions\r\n      let skip = false;\r\n      for (const fileExclusion of this.file_exclusions) {\r\n        if(files[i].path.includes(fileExclusion)) {\r\n          skip = true;\r\n          this.log_exclusion(fileExclusion);\r\n          // break out of loop\r\n          break;\r\n        }\r\n      }\r\n      if(skip) {\r\n        continue; // to next file\r\n      }\r\n      // check if file is open\r\n      if(open_files.indexOf(files[i]) > -1) {\r\n        // console.log(\"skipping file (open)\");\r\n        continue;\r\n      }\r\n      try {\r\n        // push promise to batch_promises\r\n        batch_promises.push(this.get_file_embeddings(files[i], false));\r\n      } catch (error) {\r\n        console.log(error);\r\n      }\r\n      // if batch_promises length is 10\r\n      if(batch_promises.length > 3) {\r\n        // wait for all promises to resolve\r\n        await Promise.all(batch_promises);\r\n        // clear batch_promises\r\n        batch_promises = [];\r\n      }\r\n\r\n      // save embeddings JSON to file every 100 files to save progress on bulk embedding\r\n      if(i > 0 && i % 100 === 0) {\r\n        await this.save_embeddings_to_file();\r\n      }\r\n    }\r\n    // wait for all promises to resolve\r\n    await Promise.all(batch_promises);\r\n    // write embeddings JSON to file\r\n    await this.save_embeddings_to_file();\r\n    // if render_log.failed_embeddings then update failed_embeddings.txt\r\n    if(this.render_log.failed_embeddings.length > 0) {\r\n      await this.save_failed_embeddings();\r\n    }\r\n  }\r\n\r\n  async save_embeddings_to_file(force=false) {\r\n    if(!this.has_new_embeddings){\r\n      return;\r\n    }\r\n    // console.log(\"new embeddings, saving to file\");\r\n    if(!force) {\r\n      // prevent excessive writes to embeddings file by waiting 1 minute before writing\r\n      if(this.save_timeout) {\r\n        clearTimeout(this.save_timeout);\r\n        this.save_timeout = null;\r\n      }\r\n      this.save_timeout = setTimeout(() => {\r\n        // console.log(\"writing embeddings to file\");\r\n        this.save_embeddings_to_file(true);\r\n        // clear timeout\r\n        if(this.save_timeout) {\r\n          clearTimeout(this.save_timeout);\r\n          this.save_timeout = null;\r\n        }\r\n      }, 30000);\r\n      console.log(\"scheduled save\");\r\n      return;\r\n    }\r\n\r\n    try{\r\n      // use smart_vec_lite\r\n      await this.smart_vec_lite.save();\r\n      this.has_new_embeddings = false;\r\n    }catch(error){\r\n      console.log(error);\r\n      new Obsidian.Notice(\"Smart Connections: \"+error.message);\r\n    }\r\n\r\n  }\r\n  // save failed embeddings to file from render_log.failed_embeddings\r\n  async save_failed_embeddings () {\r\n    // write failed_embeddings to file one line per failed embedding\r\n    let failed_embeddings = [];\r\n    // if file already exists then read it\r\n    const failed_embeddings_file_exists = await this.app.vault.adapter.exists(\".smart-connections/failed-embeddings.txt\");\r\n    if(failed_embeddings_file_exists) {\r\n      failed_embeddings = await this.app.vault.adapter.read(\".smart-connections/failed-embeddings.txt\");\r\n      // split failed_embeddings into array\r\n      failed_embeddings = failed_embeddings.split(\"\\r\\n\");\r\n    }\r\n    // merge failed_embeddings with render_log.failed_embeddings\r\n    failed_embeddings = failed_embeddings.concat(this.render_log.failed_embeddings);\r\n    // remove duplicates\r\n    failed_embeddings = [...new Set(failed_embeddings)];\r\n    // sort failed_embeddings array alphabetically\r\n    failed_embeddings.sort();\r\n    // convert failed_embeddings array to string\r\n    failed_embeddings = failed_embeddings.join(\"\\r\\n\");\r\n    // write failed_embeddings to file\r\n    await this.app.vault.adapter.write(\".smart-connections/failed-embeddings.txt\", failed_embeddings);\r\n    // reload failed_embeddings to prevent retrying failed files until explicitly requested\r\n    await this.load_failed_files();\r\n  }\r\n\r\n  // load failed files from failed-embeddings.txt\r\n  async load_failed_files () {\r\n    // check if failed-embeddings.txt exists\r\n    const failed_embeddings_file_exists = await this.app.vault.adapter.exists(\".smart-connections/failed-embeddings.txt\");\r\n    if(!failed_embeddings_file_exists) {\r\n      this.settings.failed_files = [];\r\n      console.log(\"No failed files.\");\r\n      return;\r\n    }\r\n    // read failed-embeddings.txt\r\n    const failed_embeddings = await this.app.vault.adapter.read(\".smart-connections/failed-embeddings.txt\");\r\n    // split failed_embeddings into array and remove empty lines\r\n    const failed_embeddings_array = failed_embeddings.split(\"\\r\\n\");\r\n    // split at '#' and reduce into unique file paths\r\n    const failed_files = failed_embeddings_array.map(embedding => embedding.split(\"#\")[0]).reduce((unique, item) => unique.includes(item) ? unique : [...unique, item], []);\r\n    // return failed_files\r\n    this.settings.failed_files = failed_files;\r\n    // console.log(failed_files);\r\n  }\r\n  // retry failed embeddings\r\n  async retry_failed_files () {\r\n    // remove failed files from failed_files\r\n    this.settings.failed_files = [];\r\n    // if failed-embeddings.txt exists then delete it\r\n    const failed_embeddings_file_exists = await this.app.vault.adapter.exists(\".smart-connections/failed-embeddings.txt\");\r\n    if(failed_embeddings_file_exists) {\r\n      await this.app.vault.adapter.remove(\".smart-connections/failed-embeddings.txt\");\r\n    }\r\n    // run get all embeddings\r\n    await this.get_all_embeddings();\r\n  }\r\n\r\n\r\n  // add .smart-connections to .gitignore to prevent issues with large, frequently updated embeddings file(s)\r\n  async add_to_gitignore() {\r\n    if(!(await this.app.vault.adapter.exists(\".gitignore\"))) {\r\n      return; // if .gitignore doesn't exist then don't add .smart-connections to .gitignore\r\n    }\r\n    let gitignore_file = await this.app.vault.adapter.read(\".gitignore\");\r\n    // if .smart-connections not in .gitignore\r\n    if (gitignore_file.indexOf(\".smart-connections\") < 0) {\r\n      // add .smart-connections to .gitignore\r\n      let add_to_gitignore = \"\\n\\n# Ignore Smart Connections folder because embeddings file is large and updated frequently\";\r\n      add_to_gitignore += \"\\n.smart-connections\";\r\n      await this.app.vault.adapter.write(\".gitignore\", gitignore_file + add_to_gitignore);\r\n      console.log(\"added .smart-connections to .gitignore\");\r\n    }\r\n  }\r\n\r\n  // force refresh embeddings file but first rename existing embeddings file to .smart-connections/embeddings-YYYY-MM-DD.json\r\n  async force_refresh_embeddings_file() {\r\n    new Obsidian.Notice(\"Smart Connections: embeddings file Force Refreshed, making new connections...\");\r\n    // force refresh\r\n    await this.smart_vec_lite.force_refresh();\r\n    // trigger making new connections\r\n    await this.get_all_embeddings();\r\n    this.output_render_log();\r\n    new Obsidian.Notice(\"Smart Connections: embeddings file Force Refreshed, new connections made.\");\r\n  }\r\n\r\n  // get embeddings for embed_input\r\n  async get_file_embeddings(curr_file, save=true) {\r\n    // let batch_promises = [];\r\n    let req_batch = [];\r\n    let blocks = [];\r\n    // initiate curr_file_key from md5(curr_file.path)\r\n    const curr_file_key = md5(curr_file.path);\r\n    // intiate file_file_embed_input by removing .md and converting file path to breadcrumbs (\" > \")\r\n    let file_embed_input = curr_file.path.replace(\".md\", \"\");\r\n    file_embed_input = file_embed_input.replace(/\\//g, \" > \");\r\n    // embed on file.name/title only if path_only path matcher specified in settings\r\n    let path_only = false;\r\n    for(let j = 0; j < this.path_only.length; j++) {\r\n      if(curr_file.path.indexOf(this.path_only[j]) > -1) {\r\n        path_only = true;\r\n        console.log(\"title only file with matcher: \" + this.path_only[j]);\r\n        // break out of loop\r\n        break;\r\n      }\r\n    }\r\n    // return early if path_only\r\n    if(path_only) {\r\n      req_batch.push([curr_file_key, file_embed_input, {\r\n        mtime: curr_file.stat.mtime,\r\n        path: curr_file.path,\r\n      }]);\r\n      await this.get_embeddings_batch(req_batch);\r\n      return;\r\n    }\r\n    /**\r\n     * BEGIN Canvas file type Embedding\r\n     */\r\n    if(curr_file.extension === \"canvas\") {\r\n      // get file contents and parse as JSON\r\n      const canvas_contents = await this.app.vault.cachedRead(curr_file);\r\n      if((typeof canvas_contents === \"string\") && (canvas_contents.indexOf(\"nodes\") > -1)) {\r\n        const canvas_json = JSON.parse(canvas_contents);\r\n        // for each object in nodes array\r\n        for(let j = 0; j < canvas_json.nodes.length; j++) {\r\n          // if object has text property\r\n          if(canvas_json.nodes[j].text) {\r\n            // add to file_embed_input\r\n            file_embed_input += \"\\n\" + canvas_json.nodes[j].text;\r\n          }\r\n          // if object has file property\r\n          if(canvas_json.nodes[j].file) {\r\n            // add to file_embed_input\r\n            file_embed_input += \"\\nLink: \" + canvas_json.nodes[j].file;\r\n          }\r\n        }\r\n      }\r\n      // console.log(file_embed_input);\r\n      req_batch.push([curr_file_key, file_embed_input, {\r\n        mtime: curr_file.stat.mtime,\r\n        path: curr_file.path,\r\n      }]);\r\n      await this.get_embeddings_batch(req_batch);\r\n      return;\r\n    }\r\n\r\n    /**\r\n     * BEGIN Block \"section\" embedding\r\n     */\r\n    // get file contents\r\n    const note_contents = await this.app.vault.cachedRead(curr_file);\r\n    let processed_since_last_save = 0;\r\n    const note_sections = this.block_parser(note_contents, curr_file.path);\r\n    // console.log(note_sections);\r\n    // if note has more than one section (if only one then its same as full-content)\r\n    if(note_sections.length > 1) {\r\n      // for each section in file\r\n      //console.log(\"Sections: \" + note_sections.length);\r\n      for (let j = 0; j < note_sections.length; j++) {\r\n        // get embed_input for block\r\n        const block_embed_input = note_sections[j].text;\r\n        // console.log(note_sections[j].path);\r\n        // get block key from block.path (contains both file.path and header path)\r\n        const block_key = md5(note_sections[j].path);\r\n        blocks.push(block_key);\r\n        // skip if length of block_embed_input same as length of embeddings[block_key].meta.size\r\n        // TODO consider rounding to nearest 10 or 100 for fuzzy matching\r\n        if (this.smart_vec_lite.get_size(block_key) === block_embed_input.length) {\r\n          // log skipping file\r\n          // console.log(\"skipping block (len)\");\r\n          continue;\r\n        }\r\n        // add hash to blocks to prevent empty blocks triggering full-file embedding\r\n        // skip if embeddings key already exists and block mtime is greater than or equal to file mtime\r\n        if(this.smart_vec_lite.mtime_is_current(block_key, curr_file.stat.mtime)) {\r\n          // log skipping file\r\n          // console.log(\"skipping block (mtime)\");\r\n          continue;\r\n        }\r\n        // skip if hash is present in embeddings and hash of block_embed_input is equal to hash in embeddings\r\n        const block_hash = md5(block_embed_input.trim());\r\n        if(this.smart_vec_lite.get_hash(block_key) === block_hash) {\r\n          // log skipping file\r\n          // console.log(\"skipping block (hash)\");\r\n          continue;\r\n        }\r\n\r\n        // create req_batch for batching requests\r\n        req_batch.push([block_key, block_embed_input, {\r\n          // oldmtime: curr_file.stat.mtime,\r\n          // get current datetime as unix timestamp\r\n          mtime: Date.now(),\r\n          hash: block_hash,\r\n          parent: curr_file_key,\r\n          path: note_sections[j].path,\r\n          size: block_embed_input.length,\r\n        }]);\r\n        if(req_batch.length > 9) {\r\n          // add batch to batch_promises\r\n          await this.get_embeddings_batch(req_batch);\r\n          processed_since_last_save += req_batch.length;\r\n          // log embedding\r\n          // console.log(\"embedding: \" + curr_file.path);\r\n          if (processed_since_last_save >= 30) {\r\n            // write embeddings JSON to file\r\n            await this.save_embeddings_to_file();\r\n            // reset processed_since_last_save\r\n            processed_since_last_save = 0;\r\n          }\r\n          // reset req_batch\r\n          req_batch = [];\r\n        }\r\n      }\r\n    }\r\n    // if req_batch is not empty\r\n    if(req_batch.length > 0) {\r\n      // process remaining req_batch\r\n      await this.get_embeddings_batch(req_batch);\r\n      req_batch = [];\r\n      processed_since_last_save += req_batch.length;\r\n    }\r\n\r\n    /**\r\n     * BEGIN File \"full note\" embedding\r\n     */\r\n\r\n    // if file length is less than ~8000 tokens use full file contents\r\n    // else if file length is greater than 8000 tokens build file_embed_input from file headings\r\n    file_embed_input += `:\\n`;\r\n    /**\r\n     * TODO: improve/refactor the following \"large file reduce to headings\" logic\r\n     */\r\n    if(note_contents.length < MAX_EMBED_STRING_LENGTH) {\r\n      file_embed_input += note_contents\r\n    }else{\r\n      const note_meta_cache = this.app.metadataCache.getFileCache(curr_file);\r\n      // for each heading in file\r\n      if(typeof note_meta_cache.headings === \"undefined\") {\r\n        // console.log(\"no headings found, using first chunk of file instead\");\r\n        file_embed_input += note_contents.substring(0, MAX_EMBED_STRING_LENGTH);\r\n      }else{\r\n        let note_headings = \"\";\r\n        for (let j = 0; j < note_meta_cache.headings.length; j++) {\r\n          // get heading level\r\n          const heading_level = note_meta_cache.headings[j].level;\r\n          // get heading text\r\n          const heading_text = note_meta_cache.headings[j].heading;\r\n          // build markdown heading\r\n          let md_heading = \"\";\r\n          for (let k = 0; k < heading_level; k++) {\r\n            md_heading += \"#\";\r\n          }\r\n          // add heading to note_headings\r\n          note_headings += `${md_heading} ${heading_text}\\n`;\r\n        }\r\n        //console.log(note_headings);\r\n        file_embed_input += note_headings\r\n        if(file_embed_input.length > MAX_EMBED_STRING_LENGTH) {\r\n          file_embed_input = file_embed_input.substring(0, MAX_EMBED_STRING_LENGTH);\r\n        }\r\n      }\r\n    }\r\n    // skip embedding full file if blocks is not empty and all hashes are present in embeddings\r\n    // better than hashing file_embed_input because more resilient to inconsequential changes (whitespace between headings)\r\n    const file_hash = md5(file_embed_input.trim());\r\n    const existing_hash = this.smart_vec_lite.get_hash(curr_file_key);\r\n    if(existing_hash && (file_hash === existing_hash)) {\r\n      // console.log(\"skipping file (hash): \" + curr_file.path);\r\n      this.update_render_log(blocks, file_embed_input);\r\n      return;\r\n    };\r\n\r\n    // if not already skipping and blocks are present\r\n    const existing_blocks = this.smart_vec_lite.get_children(curr_file_key);\r\n    let existing_has_all_blocks = true;\r\n    if(existing_blocks && Array.isArray(existing_blocks) && (blocks.length > 0)) {\r\n      // if all blocks are in existing_blocks then skip (allows deletion of small blocks without triggering full file embedding)\r\n      for (let j = 0; j < blocks.length; j++) {\r\n        if(existing_blocks.indexOf(blocks[j]) === -1) {\r\n          existing_has_all_blocks = false;\r\n          break;\r\n        }\r\n      }\r\n    }\r\n    // if existing has all blocks then check file size for delta\r\n    if(existing_has_all_blocks){\r\n      // get current note file size\r\n      const curr_file_size = curr_file.stat.size;\r\n      // get file size from embeddings\r\n      const prev_file_size = this.smart_vec_lite.get_size(curr_file_key);\r\n      if (prev_file_size) {\r\n        // if curr file size is less than 10% different from prev file size\r\n        const file_delta_pct = Math.round((Math.abs(curr_file_size - prev_file_size) / curr_file_size) * 100);\r\n        if(file_delta_pct < 10) {\r\n          // skip embedding\r\n          // console.log(\"skipping file (size) \" + curr_file.path);\r\n          this.render_log.skipped_low_delta[curr_file.name] = file_delta_pct + \"%\";\r\n          this.update_render_log(blocks, file_embed_input);\r\n          return;\r\n        }\r\n      }\r\n    }\r\n    let meta = {\r\n      mtime: curr_file.stat.mtime,\r\n      hash: file_hash,\r\n      path: curr_file.path,\r\n      size: curr_file.stat.size,\r\n      children: blocks,\r\n    };\r\n    // batch_promises.push(this.get_embeddings(curr_file_key, file_embed_input, meta));\r\n    req_batch.push([curr_file_key, file_embed_input, meta]);\r\n    // send batch request\r\n    await this.get_embeddings_batch(req_batch);\r\n\r\n    // log embedding\r\n    // console.log(\"embedding: \" + curr_file.path);\r\n    if (save) {\r\n      // write embeddings JSON to file\r\n      await this.save_embeddings_to_file();\r\n    }\r\n\r\n  }\r\n\r\n  update_render_log(blocks, file_embed_input) {\r\n    if (blocks.length > 0) {\r\n      // multiply by 2 because implies we saved token spending on blocks(sections), too\r\n      this.render_log.tokens_saved_by_cache += file_embed_input.length / 2;\r\n    } else {\r\n      // calc tokens saved by cache: divide by 4 for token estimate\r\n      this.render_log.tokens_saved_by_cache += file_embed_input.length / 4;\r\n    }\r\n  }\r\n\r\n  async get_embeddings_batch(req_batch) {\r\n    console.log(\"get_embeddings_batch\");\r\n    // if req_batch is empty then return\r\n    if(req_batch.length === 0) return;\r\n    // create arrary of embed_inputs from req_batch[i][1]\r\n    const embed_inputs = req_batch.map((req) => req[1]);\r\n    // request embeddings from embed_inputs\r\n    const requestResults = await this.request_embedding_from_input(embed_inputs);\r\n    // if requestResults is null then return\r\n    if(!requestResults) {\r\n      console.log(\"failed embedding batch\");\r\n      // log failed file names to render_log\r\n      this.render_log.failed_embeddings = [...this.render_log.failed_embeddings, ...req_batch.map((req) => req[2].path)];\r\n      return;\r\n    }\r\n    // if requestResults is not null\r\n    if(requestResults){\r\n      this.has_new_embeddings = true;\r\n      // add embedding key to render_log\r\n      if(this.settings.log_render){\r\n        if(this.settings.log_render_files){\r\n          this.render_log.files = [...this.render_log.files, ...req_batch.map((req) => req[2].path)];\r\n        }\r\n        this.render_log.new_embeddings += req_batch.length;\r\n        // add token usage to render_log\r\n        this.render_log.token_usage += requestResults.usage.total_tokens;\r\n      }\r\n      // console.log(requestResults.data.length);\r\n      // loop through requestResults.data\r\n      for(let i = 0; i < requestResults.data.length; i++) {\r\n        const vec = requestResults.data[i].embedding;\r\n        const index = requestResults.data[i].index;\r\n        if(vec) {\r\n          const key = req_batch[index][0];\r\n          const meta = req_batch[index][2];\r\n          this.smart_vec_lite.save_embedding(key, vec, meta);\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  async request_embedding_from_input(embed_input, retries = 0) {\r\n    // (FOR TESTING) test fail process by forcing fail\r\n    // return null;\r\n    // check if embed_input is a string\r\n    // if(typeof embed_input !== \"string\") {\r\n    //   console.log(\"embed_input is not a string\");\r\n    //   return null;\r\n    // }\r\n    // check if embed_input is empty\r\n    if(embed_input.length === 0) {\r\n      console.log(\"embed_input is empty\");\r\n      return null;\r\n    }\r\n    const usedParams = {\r\n      model: \"text-embedding-ada-002\",\r\n      input: embed_input,\r\n    };\r\n    // console.log(this.settings.api_key);\r\n    const reqParams = {\r\n      url: `https://api.openai.com/v1/embeddings`,\r\n      method: \"POST\",\r\n      body: JSON.stringify(usedParams),\r\n      headers: {\r\n        \"Content-Type\": \"application/json\",\r\n        \"Authorization\": `Bearer ${this.settings.api_key}`\r\n      }\r\n    };\r\n    let resp;\r\n    try {\r\n      resp = await (0, Obsidian.request)(reqParams)\r\n      return JSON.parse(resp);\r\n    } catch (error) {\r\n      // retry request if error is 429\r\n      if((error.status === 429) && (retries < 3)) {\r\n        retries++;\r\n        // exponential backoff\r\n        const backoff = Math.pow(retries, 2);\r\n        console.log(`retrying request (429) in ${backoff} seconds...`);\r\n        await new Promise(r => setTimeout(r, 1000 * backoff));\r\n        return await this.request_embedding_from_input(embed_input, retries);\r\n      }\r\n      // log full error to console\r\n      console.log(resp);\r\n      // console.log(\"first line of embed: \" + embed_input.substring(0, embed_input.indexOf(\"\\n\")));\r\n      // console.log(\"embed input length: \"+ embed_input.length);\r\n      // if(Array.isArray(embed_input)) {\r\n      //   console.log(embed_input.map((input) => input.length));\r\n      // }\r\n      // console.log(\"erroneous embed input: \" + embed_input);\r\n      console.log(error);\r\n      // console.log(usedParams);\r\n      // console.log(usedParams.input.length);\r\n      return null;\r\n    }\r\n  }\r\n  async test_api_key() {\r\n    const embed_input = \"This is a test of the OpenAI API.\";\r\n    const resp = await this.request_embedding_from_input(embed_input);\r\n    if(resp && resp.usage) {\r\n      console.log(\"API key is valid\");\r\n      return true;\r\n    }else{\r\n      console.log(\"API key is invalid\");\r\n      return false;\r\n    }\r\n  }\r\n\r\n\r\n  output_render_log() {\r\n    // if settings.log_render is true\r\n    if(this.settings.log_render) {\r\n      if (this.render_log.new_embeddings === 0) {\r\n        return;\r\n      }else{\r\n        // pretty print this.render_log to console\r\n        console.log(JSON.stringify(this.render_log, null, 2));\r\n      }\r\n    }\r\n\r\n    // clear render_log\r\n    this.render_log = {};\r\n    this.render_log.deleted_embeddings = 0;\r\n    this.render_log.exclusions_logs = {};\r\n    this.render_log.failed_embeddings = [];\r\n    this.render_log.files = [];\r\n    this.render_log.new_embeddings = 0;\r\n    this.render_log.skipped_low_delta = {};\r\n    this.render_log.token_usage = 0;\r\n    this.render_log.tokens_saved_by_cache = 0;\r\n  }\r\n\r\n  // find connections by most similar to current note by cosine similarity\r\n  async find_note_connections(current_note=null) {\r\n    // md5 of current note path\r\n    const curr_key = md5(current_note.path);\r\n    // if in this.nearest_cache then set to nearest\r\n    // else get nearest\r\n    let nearest = [];\r\n    if(this.nearest_cache[curr_key]) {\r\n      nearest = this.nearest_cache[curr_key];\r\n      // console.log(\"nearest from cache\");\r\n    }else{\r\n      // skip files where path contains any exclusions\r\n      for(let j = 0; j < this.file_exclusions.length; j++) {\r\n        if(current_note.path.indexOf(this.file_exclusions[j]) > -1) {\r\n          this.log_exclusion(this.file_exclusions[j]);\r\n          // break out of loop and finish here\r\n          return \"excluded\";\r\n        }\r\n      }\r\n      // get all embeddings\r\n      // await this.get_all_embeddings();\r\n      // wrap get all in setTimeout to allow for UI to update\r\n      setTimeout(() => {\r\n        this.get_all_embeddings()\r\n      }, 3000);\r\n      // get from cache if mtime is same and values are not empty\r\n      if(this.smart_vec_lite.mtime_is_current(curr_key, current_note.stat.mtime)) {\r\n        // skipping get file embeddings because nothing has changed\r\n        // console.log(\"find_note_connections - skipping file (mtime)\");\r\n      }else{\r\n        // get file embeddings\r\n        await this.get_file_embeddings(current_note);\r\n      }\r\n      // get current note embedding vector\r\n      const vec = this.smart_vec_lite.get_vec(curr_key);\r\n      if(!vec) {\r\n        return \"Error getting embeddings for: \"+current_note.path;\r\n      }\r\n\r\n      // compute cosine similarity between current note and all other notes via embeddings\r\n      nearest = this.smart_vec_lite.nearest(vec, {\r\n        skip_key: curr_key,\r\n        skip_sections: this.settings.skip_sections,\r\n      });\r\n\r\n      // save to this.nearest_cache\r\n      this.nearest_cache[curr_key] = nearest;\r\n    }\r\n\r\n    // return array sorted by cosine similarity\r\n    return nearest;\r\n  }\r\n\r\n  // create render_log object of exlusions with number of times skipped as value\r\n  log_exclusion(exclusion) {\r\n    // increment render_log for skipped file\r\n    this.render_log.exclusions_logs[exclusion] = (this.render_log.exclusions_logs[exclusion] || 0) + 1;\r\n  }\r\n\r\n\r\n  block_parser(markdown, file_path){\r\n    // if this.settings.skip_sections is true then return empty array\r\n    if(this.settings.skip_sections) {\r\n      return [];\r\n    }\r\n    // split the markdown into lines\r\n    const lines = markdown.split('\\n');\r\n    // initialize the blocks array\r\n    let blocks = [];\r\n    // current headers array\r\n    let currentHeaders = [];\r\n    // remove .md file extension and convert file_path to breadcrumb formatting\r\n    const file_breadcrumbs = file_path.replace('.md', '').replace(/\\//g, ' > ');\r\n    // initialize the block string\r\n    let block = '';\r\n    let block_headings = '';\r\n    let block_path = file_path;\r\n\r\n    let last_heading_line = 0;\r\n    let i = 0;\r\n    let block_headings_list = [];\r\n    // loop through the lines\r\n    for (i = 0; i < lines.length; i++) {\r\n      // get the line\r\n      const line = lines[i];\r\n      // if line does not start with #\r\n      // or if line starts with # and second character is a word or number indicating a \"tag\"\r\n      // then add to block\r\n      if (!line.startsWith('#') || (['#',' '].indexOf(line[1]) < 0)){\r\n        // skip if line is empty\r\n        if(line === '') continue;\r\n        // skip if line is empty bullet or checkbox\r\n        if(['- ', '- [ ] '].indexOf(line) > -1) continue;\r\n        // if currentHeaders is empty skip (only blocks with headers, otherwise block.path conflicts with file.path)\r\n        if(currentHeaders.length === 0) continue;\r\n        // add line to block\r\n        block += \"\\n\" + line;\r\n        continue;\r\n      }\r\n      /**\r\n       * BEGIN Heading parsing\r\n       * - likely a heading if made it this far\r\n       */\r\n      last_heading_line = i;\r\n      // push the current block to the blocks array unless last line was a also a header\r\n      if(i > 0 && (last_heading_line !== (i-1)) && (block.indexOf(\"\\n\") > -1) && this.validate_headings(block_headings)) {\r\n        output_block();\r\n      }\r\n      // get the header level\r\n      const level = line.split('#').length - 1;\r\n      // remove any headers from the current headers array that are higher than the current header level\r\n      currentHeaders = currentHeaders.filter(header => header.level < level);\r\n      // add header and level to current headers array\r\n      // trim the header to remove \"#\" and any trailing spaces\r\n      currentHeaders.push({header: line.replace(/#/g, '').trim(), level: level});\r\n      // initialize the block breadcrumbs with file.path the current headers\r\n      block = file_breadcrumbs;\r\n      block += \": \" + currentHeaders.map(header => header.header).join(' > ');\r\n      block_headings = \"#\"+currentHeaders.map(header => header.header).join('#');\r\n      // if block_headings is already in block_headings_list then add a number to the end\r\n      if(block_headings_list.indexOf(block_headings) > -1) {\r\n        let count = 1;\r\n        while(block_headings_list.indexOf(`${block_headings}{${count}}`) > -1) {\r\n          count++;\r\n        }\r\n        block_headings = `${block_headings}{${count}}`;\r\n      }\r\n      block_headings_list.push(block_headings);\r\n      block_path = file_path + block_headings;\r\n    }\r\n    // handle remaining after loop\r\n    if((last_heading_line !== (i-1)) && (block.indexOf(\"\\n\") > -1) && this.validate_headings(block_headings)) output_block();\r\n    // remove any blocks that are too short (length < 50)\r\n    blocks = blocks.filter(b => b.length > 50);\r\n    // console.log(blocks);\r\n    // return the blocks array\r\n    return blocks;\r\n\r\n    function output_block() {\r\n      // breadcrumbs length (first line of block)\r\n      const breadcrumbs_length = block.indexOf(\"\\n\") + 1;\r\n      const block_length = block.length - breadcrumbs_length;\r\n      // trim block to max length\r\n      if (block.length > MAX_EMBED_STRING_LENGTH) {\r\n        block = block.substring(0, MAX_EMBED_STRING_LENGTH);\r\n      }\r\n      blocks.push({ text: block.trim(), path: block_path, length: block_length });\r\n    }\r\n  }\r\n  // reverse-retrieve block given path\r\n  async block_retriever(path, limits={}) {\r\n    limits = {\r\n      lines: null,\r\n      chars_per_line: null,\r\n      max_chars: null,\r\n      ...limits\r\n    }\r\n    // return if no # in path\r\n    if (path.indexOf('#') < 0) {\r\n      console.log(\"not a block path: \"+path);\r\n      return false;\r\n    }\r\n    let block = [];\r\n    let block_headings = path.split('#').slice(1);\r\n    // if path ends with number in curly braces\r\n    let heading_occurrence = 0;\r\n    if(block_headings[block_headings.length-1].indexOf('{') > -1) {\r\n      // get the occurrence number\r\n      heading_occurrence = parseInt(block_headings[block_headings.length-1].split('{')[1].replace('}', ''));\r\n      // remove the occurrence from the last heading\r\n      block_headings[block_headings.length-1] = block_headings[block_headings.length-1].split('{')[0];\r\n    }\r\n    let currentHeaders = [];\r\n    let occurrence_count = 0;\r\n    let begin_line = 0;\r\n    let i = 0;\r\n    // get file path from path\r\n    const file_path = path.split('#')[0];\r\n    // get file\r\n    const file = this.app.vault.getAbstractFileByPath(file_path);\r\n    if(!(file instanceof Obsidian.TFile)) {\r\n      console.log(\"not a file: \"+file_path);\r\n      return false;\r\n    }\r\n    // get file contents\r\n    const file_contents = await this.app.vault.cachedRead(file);\r\n    // split the file contents into lines\r\n    const lines = file_contents.split('\\n');\r\n    // loop through the lines\r\n    let is_code = false;\r\n    for (i = 0; i < lines.length; i++) {\r\n      // get the line\r\n      const line = lines[i];\r\n      // if line begins with three backticks then toggle is_code\r\n      if(line.indexOf('```') === 0) {\r\n        is_code = !is_code;\r\n      }\r\n      // if is_code is true then add line with preceding tab and continue\r\n      if(is_code) {\r\n        continue;\r\n      }\r\n      // skip if line is empty bullet or checkbox\r\n      if(['- ', '- [ ] '].indexOf(line) > -1) continue;\r\n      // if line does not start with #\r\n      // or if line starts with # and second character is a word or number indicating a \"tag\"\r\n      // then continue to next line\r\n      if (!line.startsWith('#') || (['#',' '].indexOf(line[1]) < 0)){\r\n        continue;\r\n      }\r\n      /**\r\n       * BEGIN Heading parsing\r\n       * - likely a heading if made it this far\r\n       */\r\n      // get the heading text\r\n      const heading_text = line.replace(/#/g, '').trim();\r\n      // continue if heading text is not in block_headings\r\n      const heading_index = block_headings.indexOf(heading_text);\r\n      if (heading_index < 0) continue;\r\n      // if currentHeaders.length !== heading_index then we have a mismatch\r\n      if (currentHeaders.length !== heading_index) continue;\r\n      // push the heading text to the currentHeaders array\r\n      currentHeaders.push(heading_text);\r\n      // if currentHeaders.length === block_headings.length then we have a match\r\n      if (currentHeaders.length === block_headings.length) {\r\n        // if heading_occurrence is defined then increment occurrence_count\r\n        if(heading_occurrence === 0) {\r\n          // set begin_line to i + 1\r\n          begin_line = i + 1;\r\n          break; // break out of loop\r\n        }\r\n        // if occurrence_count !== heading_occurrence then continue\r\n        if(occurrence_count === heading_occurrence){\r\n          begin_line = i + 1;\r\n          break; // break out of loop\r\n        }\r\n        occurrence_count++;\r\n        // reset currentHeaders\r\n        currentHeaders.pop();\r\n        continue;\r\n      }\r\n    }\r\n    // if no begin_line then return false\r\n    if (begin_line === 0) return false;\r\n    // iterate through lines starting at begin_line\r\n    is_code = false;\r\n    // character accumulator\r\n    let char_count = 0;\r\n    for (i = begin_line; i < lines.length; i++) {\r\n      if((typeof line_limit === \"number\") && (block.length > line_limit)){\r\n        block.push(\"...\");\r\n        break; // ends when line_limit is reached\r\n      }\r\n      let line = lines[i];\r\n      if ((line.indexOf('#') === 0) && (['#',' '].indexOf(line[1]) !== -1)){\r\n        break; // ends when encountering next header\r\n      }\r\n      // DEPRECATED: should be handled by new_line+char_count check (happens in previous iteration)\r\n      // if char_count is greater than limit.max_chars, skip\r\n      if (limits.max_chars && char_count > limits.max_chars) {\r\n        block.push(\"...\");\r\n        break;\r\n      }\r\n      // if new_line + char_count is greater than limit.max_chars, skip\r\n      if (limits.max_chars && ((line.length + char_count) > limits.max_chars)) {\r\n        const max_new_chars = limits.max_chars - char_count;\r\n        line = line.slice(0, max_new_chars) + \"...\";\r\n        break;\r\n      }\r\n      // validate/format\r\n      // if line is empty, skip\r\n      if (line.length === 0) continue;\r\n      // limit length of line to N characters\r\n      if (limits.chars_per_line && line.length > limits.chars_per_line) {\r\n        line = line.slice(0, limits.chars_per_line) + \"...\";\r\n      }\r\n      // if line is a code block, skip\r\n      if (line.startsWith(\"```\")) {\r\n        is_code = !is_code;\r\n        continue;\r\n      }\r\n      if (is_code){\r\n        // add tab to beginning of line\r\n        line = \"\\t\"+line;\r\n      }\r\n      // add line to block\r\n      block.push(line);\r\n      // increment char_count\r\n      char_count += line.length;\r\n    }\r\n    // close code block if open\r\n    if (is_code) {\r\n      block.push(\"```\");\r\n    }\r\n    return block.join(\"\\n\").trim();\r\n  }\r\n\r\n  // retrieve a file from the vault\r\n  async file_retriever(link, limits={}) {\r\n    limits = {\r\n      lines: null,\r\n      max_chars: null,\r\n      chars_per_line: null,\r\n      ...limits\r\n    };\r\n    const this_file = this.app.vault.getAbstractFileByPath(link);\r\n    // if file is not found, skip\r\n    if (!(this_file instanceof Obsidian.TAbstractFile)) return false;\r\n    // use cachedRead to get the first 10 lines of the file\r\n    const file_content = await this.app.vault.cachedRead(this_file);\r\n    const file_lines = file_content.split(\"\\n\");\r\n    let first_ten_lines = [];\r\n    let is_code = false;\r\n    let char_accum = 0;\r\n    const line_limit = limits.lines || file_lines.length;\r\n    for (let i = 0; first_ten_lines.length < line_limit; i++) {\r\n      let line = file_lines[i];\r\n      // if line is undefined, break\r\n      if (typeof line === 'undefined')\r\n        break;\r\n      // if line is empty, skip\r\n      if (line.length === 0)\r\n        continue;\r\n      // limit length of line to N characters\r\n      if (limits.chars_per_line && line.length > limits.chars_per_line) {\r\n        line = line.slice(0, limits.chars_per_line) + \"...\";\r\n      }\r\n      // if line is \"---\", skip\r\n      if (line === \"---\")\r\n        continue;\r\n      // skip if line is empty bullet or checkbox\r\n      if (['- ', '- [ ] '].indexOf(line) > -1)\r\n        continue;\r\n      // if line is a code block, skip\r\n      if (line.indexOf(\"```\") === 0) {\r\n        is_code = !is_code;\r\n        continue;\r\n      }\r\n      // if char_accum is greater than limit.max_chars, skip\r\n      if (limits.max_chars && char_accum > limits.max_chars) {\r\n        first_ten_lines.push(\"...\");\r\n        break;\r\n      }\r\n      if (is_code) {\r\n        // if is code, add tab to beginning of line\r\n        line = \"\\t\" + line;\r\n      }\r\n      // if line is a heading\r\n      if (line_is_heading(line)) {\r\n        // look at last line in first_ten_lines to see if it is a heading\r\n        // note: uses last in first_ten_lines, instead of look ahead in file_lines, because..\r\n        // ...next line may be excluded from first_ten_lines by previous if statements\r\n        if ((first_ten_lines.length > 0) && line_is_heading(first_ten_lines[first_ten_lines.length - 1])) {\r\n          // if last line is a heading, remove it\r\n          first_ten_lines.pop();\r\n        }\r\n      }\r\n      // add line to first_ten_lines\r\n      first_ten_lines.push(line);\r\n      // increment char_accum\r\n      char_accum += line.length;\r\n    }\r\n    // for each line in first_ten_lines, apply view-specific formatting\r\n    for (let i = 0; i < first_ten_lines.length; i++) {\r\n      // if line is a heading\r\n      if (line_is_heading(first_ten_lines[i])) {\r\n        // if this is the last line in first_ten_lines\r\n        if (i === first_ten_lines.length - 1) {\r\n          // remove the last line if it is a heading\r\n          first_ten_lines.pop();\r\n          break;\r\n        }\r\n        // remove heading syntax to improve readability in small space\r\n        first_ten_lines[i] = first_ten_lines[i].replace(/#+/, \"\");\r\n        first_ten_lines[i] = `\\n${first_ten_lines[i]}:`;\r\n      }\r\n    }\r\n    // join first ten lines into string\r\n    first_ten_lines = first_ten_lines.join(\"\\n\");\r\n    return first_ten_lines;\r\n  }\r\n\r\n  // iterate through blocks and skip if block_headings contains this.header_exclusions\r\n  validate_headings(block_headings) {\r\n    let valid = true;\r\n    if (this.header_exclusions.length > 0) {\r\n      for (let k = 0; k < this.header_exclusions.length; k++) {\r\n        if (block_headings.indexOf(this.header_exclusions[k]) > -1) {\r\n          valid = false;\r\n          this.log_exclusion(\"heading: \"+this.header_exclusions[k]);\r\n          break;\r\n        }\r\n      }\r\n    }\r\n    return valid;\r\n  }\r\n  // render \"Smart Connections\" text fixed in the bottom right corner\r\n  render_brand(container, location=\"default\") {\r\n    // if location is all then get Object.keys(this.sc_branding) and call this function for each\r\n    if (container === \"all\") {\r\n      const locations = Object.keys(this.sc_branding);\r\n      for (let i = 0; i < locations.length; i++) {\r\n        this.render_brand(this.sc_branding[locations[i]], locations[i]);\r\n      }\r\n      return;\r\n    }\r\n    // brand container\r\n    this.sc_branding[location] = container;\r\n    // if this.sc_branding[location] contains child with class \"sc-brand\", remove it\r\n    if (this.sc_branding[location].querySelector(\".sc-brand\")) {\r\n      this.sc_branding[location].querySelector(\".sc-brand\").remove();\r\n    }\r\n    const brand_container = this.sc_branding[location].createEl(\"div\", { cls: \"sc-brand\" });\r\n    // add text\r\n    // add SVG signal icon using getIcon\r\n    Obsidian.setIcon(brand_container, \"smart-connections\");\r\n    const brand_p = brand_container.createEl(\"p\");\r\n    let text = \"Smart Connections\";\r\n    let attr = {};\r\n    // if update available, change text to \"Update Available\"\r\n    if (this.update_available) {\r\n      text = \"Update Available\";\r\n      attr = {\r\n        style: \"font-weight: 700;\"\r\n      };\r\n    }\r\n    brand_p.createEl(\"a\", {\r\n      cls: \"\",\r\n      text: text,\r\n      href: \"https://github.com/brianpetro/obsidian-smart-connections/discussions\",\r\n      target: \"_blank\",\r\n      attr: attr\r\n    });\r\n  }\r\n\r\n\r\n  // create list of nearest notes\r\n  async update_results(container, nearest) {\r\n    let list;\r\n    // check if list exists\r\n    if((container.children.length > 1) && (container.children[1].classList.contains(\"sc-list\"))){\r\n      list = container.children[1];\r\n    }\r\n    // if list exists, empty it\r\n    if (list) {\r\n      list.empty();\r\n    } else {\r\n      // create list element\r\n      list = container.createEl(\"div\", { cls: \"sc-list\" });\r\n    }\r\n    let search_result_class = \"search-result\";\r\n    // if settings expanded_view is false, add sc-collapsed class\r\n    if(!this.settings.expanded_view) search_result_class += \" sc-collapsed\";\r\n\r\n    // TODO: add option to group nearest by file\r\n    if(!this.settings.group_nearest_by_file) {\r\n      // for each nearest note\r\n      for (let i = 0; i < nearest.length; i++) {\r\n        /**\r\n         * BEGIN EXTERNAL LINK LOGIC\r\n         * if link is an object, it indicates external link\r\n         */\r\n        console.log(this);\r\n        if (typeof nearest[i].link === \"object\") {\r\n          const item = list.createEl(\"div\", { cls: \"search-result\" });\r\n          const link = item.createEl(\"a\", {\r\n            cls: \"search-result-file-title is-clickable\",\r\n            href: nearest[i].link.path,\r\n            title: nearest[i].link.title,\r\n          });\r\n          link.innerHTML = this.render_external_link_elm(nearest[i].link);\r\n          item.setAttr('draggable', 'true')\r\n          continue; // ends here for external links\r\n        }\r\n        /**\r\n         * BEGIN INTERNAL LINK LOGIC\r\n         * if link is a string, it indicates internal link\r\n         */\r\n        let file_link_text;\r\n        const file_similarity_pct = Math.round(nearest[i].similarity * 100) + \"%\";\r\n        if(this.settings.show_full_path) {\r\n          const pcs = nearest[i].link.split(\"/\");\r\n          file_link_text = pcs[pcs.length - 1];\r\n          const path = pcs.slice(0, pcs.length - 1).join(\"/\");\r\n          // file_link_text = `<small>${path} | ${file_similarity_pct}</small><br>${file_link_text}`;\r\n          file_link_text = `<small>${file_similarity_pct} | ${path} | ${file_link_text}</small>`;\r\n        }else{\r\n          file_link_text = '<small>' + file_similarity_pct + \" | \" + nearest[i].link.split(\"/\").pop() + '</small>';\r\n        }\r\n        // skip contents rendering if incompatible file type\r\n        // ex. not markdown file or contains no '.excalidraw'\r\n        if(!this.renderable_file_type(nearest[i].link)){\r\n          const item = list.createEl(\"div\", { cls: \"search-result\" });\r\n          const link = item.createEl(\"a\", {\r\n            cls: \"search-result-file-title is-clickable\",\r\n            href: nearest[i].link,\r\n          });\r\n          link.innerHTML = file_link_text;\r\n          // drag and drop\r\n          item.setAttr('draggable', 'true')\r\n          // add listeners to link\r\n          this.add_link_listeners(link, nearest[i], item);\r\n          continue;\r\n        }\r\n\r\n        // remove file extension if .md and make # into >\r\n        file_link_text = file_link_text.replace(\".md\", \"\").replace(/#/g, \" > \");\r\n        // create item\r\n        const item = list.createEl(\"div\", { cls: search_result_class });\r\n        // create span for toggle\r\n        const toggle = item.createEl(\"span\", { cls: \"is-clickable\" });\r\n        // insert right triangle svg as toggle\r\n        Obsidian.setIcon(toggle, \"right-triangle\"); // must come before adding other elms to prevent overwrite\r\n        const link = toggle.createEl(\"a\", {\r\n          cls: \"search-result-file-title\",\r\n          title: nearest[i].link,\r\n        });\r\n        link.innerHTML = file_link_text;\r\n        // add listeners to link\r\n        this.add_link_listeners(link, nearest[i], item);\r\n        toggle.addEventListener(\"click\", (event) => {\r\n          // find parent containing search-result class\r\n          let parent = event.target.parentElement;\r\n          while (!parent.classList.contains(\"search-result\")) {\r\n            parent = parent.parentElement;\r\n          }\r\n          // toggle sc-collapsed class\r\n          parent.classList.toggle(\"sc-collapsed\");\r\n        });\r\n        const contents = item.createEl(\"ul\", { cls: \"\" });\r\n        const contents_container = contents.createEl(\"li\", {\r\n          cls: \"search-result-file-title is-clickable\",\r\n          title: nearest[i].link,\r\n        });\r\n        if(nearest[i].link.indexOf(\"#\") > -1){ // is block\r\n          Obsidian.MarkdownRenderer.renderMarkdown((await this.block_retriever(nearest[i].link, {lines: 10, max_chars: 1000})), contents_container, nearest[i].link, new Obsidian.Component());\r\n        }else{ // is file\r\n          const first_ten_lines = await this.file_retriever(nearest[i].link, {lines: 10, max_chars: 1000});\r\n          if(!first_ten_lines) continue; // skip if file is empty\r\n          Obsidian.MarkdownRenderer.renderMarkdown(first_ten_lines, contents_container, nearest[i].link, new Obsidian.Component());\r\n        }\r\n        this.add_link_listeners(contents, nearest[i], item);\r\n      }\r\n      this.render_brand(container, \"block\");\r\n      return;\r\n    }\r\n\r\n    // group nearest by file\r\n    const nearest_by_file = {};\r\n    for (let i = 0; i < nearest.length; i++) {\r\n      const curr = nearest[i];\r\n      const link = curr.link;\r\n      // skip if link is an object (indicates external logic)\r\n      if (typeof link === \"object\") {\r\n        nearest_by_file[link.path] = [curr];\r\n        continue;\r\n      }\r\n      if (link.indexOf(\"#\") > -1) {\r\n        const file_path = link.split(\"#\")[0];\r\n        if (!nearest_by_file[file_path]) {\r\n          nearest_by_file[file_path] = [];\r\n        }\r\n        nearest_by_file[file_path].push(nearest[i]);\r\n      } else {\r\n        if (!nearest_by_file[link]) {\r\n          nearest_by_file[link] = [];\r\n        }\r\n        // always add to front of array\r\n        nearest_by_file[link].unshift(nearest[i]);\r\n      }\r\n    }\r\n    // for each file\r\n    const keys = Object.keys(nearest_by_file);\r\n    for (let i = 0; i < keys.length; i++) {\r\n      const file = nearest_by_file[keys[i]];\r\n      /**\r\n       * Begin external link handling\r\n       */\r\n      // if link is an object (indicates v2 logic)\r\n      if (typeof file[0].link === \"object\") {\r\n        const curr = file[0];\r\n        const meta = curr.link;\r\n        if (meta.path.startsWith(\"http\")) {\r\n          const item = list.createEl(\"div\", { cls: \"search-result\" });\r\n          const link = item.createEl(\"a\", {\r\n            cls: \"search-result-file-title is-clickable\",\r\n            href: meta.path,\r\n            title: meta.title,\r\n          });\r\n          link.innerHTML = this.render_external_link_elm(meta);\r\n          item.setAttr('draggable', 'true');\r\n          continue; // ends here for external links\r\n        }\r\n      }\r\n      /**\r\n       * Handles Internal\r\n       */\r\n      let file_link_text;\r\n      const file_similarity_pct = Math.round(file[0].similarity * 100) + \"%\";\r\n      if (this.settings.show_full_path) {\r\n        const pcs = file[0].link.split(\"/\");\r\n        file_link_text = pcs[pcs.length - 1];\r\n        const path = pcs.slice(0, pcs.length - 1).join(\"/\");\r\n        file_link_text = `<small>${path} | ${file_similarity_pct}</small><br>${file_link_text}`;\r\n      } else {\r\n        file_link_text = file[0].link.split(\"/\").pop();\r\n        // add similarity percentage\r\n        file_link_text += ' | ' + file_similarity_pct;\r\n      }\r\n\r\n\r\n\r\n      // skip contents rendering if incompatible file type\r\n      // ex. not markdown or contains '.excalidraw'\r\n      if(!this.renderable_file_type(file[0].link)) {\r\n        const item = list.createEl(\"div\", { cls: \"search-result\" });\r\n        const file_link = item.createEl(\"a\", {\r\n          cls: \"search-result-file-title is-clickable\",\r\n          title: file[0].link,\r\n        });\r\n        file_link.innerHTML = file_link_text;\r\n        // add link listeners to file link\r\n        this.add_link_listeners(file_link, file[0], item);\r\n        continue;\r\n      }\r\n\r\n\r\n      // remove file extension if .md\r\n      file_link_text = file_link_text.replace(\".md\", \"\").replace(/#/g, \" > \");\r\n      const item = list.createEl(\"div\", { cls: search_result_class });\r\n      const toggle = item.createEl(\"span\", { cls: \"is-clickable\" });\r\n      // insert right triangle svg icon as toggle button in span\r\n      Obsidian.setIcon(toggle, \"right-triangle\"); // must come before adding other elms else overwrites\r\n      const file_link = toggle.createEl(\"a\", {\r\n        cls: \"search-result-file-title\",\r\n        title: file[0].link,\r\n      });\r\n      file_link.innerHTML = file_link_text;\r\n      // add link listeners to file link\r\n      this.add_link_listeners(file_link, file[0], toggle);\r\n      toggle.addEventListener(\"click\", (event) => {\r\n        // find parent containing class search-result\r\n        let parent = event.target;\r\n        while (!parent.classList.contains(\"search-result\")) {\r\n          parent = parent.parentElement;\r\n        }\r\n        parent.classList.toggle(\"sc-collapsed\");\r\n        // TODO: if block container is empty, render markdown from block retriever\r\n      });\r\n      const file_link_list = item.createEl(\"ul\");\r\n      // for each link in file\r\n      for (let j = 0; j < file.length; j++) {\r\n        // if is a block (has # in link)\r\n        if(file[j].link.indexOf(\"#\") > -1) {\r\n          const block = file[j];\r\n          const block_link = file_link_list.createEl(\"li\", {\r\n            cls: \"search-result-file-title is-clickable\",\r\n            title: block.link,\r\n          });\r\n          // skip block context if file.length === 1 because already added\r\n          if(file.length > 1) {\r\n            const block_context = this.render_block_context(block);\r\n            const block_similarity_pct = Math.round(block.similarity * 100) + \"%\";\r\n            block_link.innerHTML = `<small>${block_context} | ${block_similarity_pct}</small>`;\r\n          }\r\n          const block_container = block_link.createEl(\"div\");\r\n          // TODO: move to rendering on expanding section (toggle collapsed)\r\n          Obsidian.MarkdownRenderer.renderMarkdown((await this.block_retriever(block.link, {lines: 10, max_chars: 1000})), block_container, block.link, new Obsidian.Component());\r\n          // add link listeners to block link\r\n          this.add_link_listeners(block_link, block, file_link_list);\r\n        }else{\r\n          // get first ten lines of file\r\n          const file_link_list = item.createEl(\"ul\");\r\n          const block_link = file_link_list.createEl(\"li\", {\r\n            cls: \"search-result-file-title is-clickable\",\r\n            title: file[0].link,\r\n          });\r\n          const block_container = block_link.createEl(\"div\");\r\n          let first_ten_lines = await this.file_retriever(file[0].link, {lines: 10, max_chars: 1000});\r\n          if(!first_ten_lines) continue; // if file not found, skip\r\n          Obsidian.MarkdownRenderer.renderMarkdown(first_ten_lines, block_container, file[0].link, new Obsidian.Component());\r\n          this.add_link_listeners(block_link, file[0], file_link_list);\r\n\r\n        }\r\n      }\r\n    }\r\n    this.render_brand(container, \"file\");\r\n  }\r\n\r\n  add_link_listeners(item, curr, list) {\r\n    item.addEventListener(\"click\", async (event) => {\r\n      await this.open_note(curr, event);\r\n    });\r\n    // drag-on\r\n    // currently only works with full-file links\r\n    item.setAttr('draggable', 'true');\r\n    item.addEventListener('dragstart', (event) => {\r\n      const dragManager = this.app.dragManager;\r\n      const file_path = curr.link.split(\"#\")[0];\r\n      const file = this.app.metadataCache.getFirstLinkpathDest(file_path, '');\r\n      const dragData = dragManager.dragFile(event, file);\r\n      // console.log(dragData);\r\n      dragManager.onDragStart(event, dragData);\r\n    });\r\n    // if curr.link contains curly braces, return (incompatible with hover-link)\r\n    if (curr.link.indexOf(\"{\") > -1) return;\r\n    // trigger hover event on link\r\n    item.addEventListener(\"mouseover\", (event) => {\r\n      this.app.workspace.trigger(\"hover-link\", {\r\n        event,\r\n        source: SMART_CONNECTIONS_VIEW_TYPE,\r\n        hoverParent: list,\r\n        targetEl: item,\r\n        linktext: curr.link,\r\n      });\r\n    });\r\n  }\r\n\r\n  // get target file from link path\r\n  // if sub-section is linked, open file and scroll to sub-section\r\n  async open_note(curr, event=null) {\r\n    let targetFile;\r\n    let heading;\r\n    if (curr.link.indexOf(\"#\") > -1) {\r\n      // remove after # from link\r\n      targetFile = this.app.metadataCache.getFirstLinkpathDest(curr.link.split(\"#\")[0], \"\");\r\n      // console.log(targetFile);\r\n      const target_file_cache = this.app.metadataCache.getFileCache(targetFile);\r\n      // console.log(target_file_cache);\r\n      // get heading\r\n      let heading_text = curr.link.split(\"#\").pop();\r\n      // if heading text contains a curly brace, get the number inside the curly braces as occurence\r\n      let occurence = 0;\r\n      if (heading_text.indexOf(\"{\") > -1) {\r\n        // get occurence\r\n        occurence = parseInt(heading_text.split(\"{\")[1].split(\"}\")[0]);\r\n        // remove occurence from heading text\r\n        heading_text = heading_text.split(\"{\")[0];\r\n      }\r\n      // get headings from file cache\r\n      const headings = target_file_cache.headings;\r\n      // get headings with the same depth and text as the link\r\n      for(let i = 0; i < headings.length; i++) {\r\n        if (headings[i].heading === heading_text) {\r\n          // if occurence is 0, set heading and break\r\n          if(occurence === 0) {\r\n            heading = headings[i];\r\n            break;\r\n          }\r\n          occurence--; // decrement occurence\r\n        }\r\n      }\r\n      // console.log(heading);\r\n    } else {\r\n      targetFile = this.app.metadataCache.getFirstLinkpathDest(curr.link, \"\");\r\n    }\r\n    let leaf;\r\n    if(event) {\r\n      // properly handle if the meta/ctrl key is pressed\r\n      const mod = Obsidian.Keymap.isModEvent(event);\r\n      // get most recent leaf\r\n      leaf = this.app.workspace.getLeaf(mod);\r\n    }else{\r\n      // get most recent leaf\r\n      leaf = this.app.workspace.getMostRecentLeaf();\r\n    }\r\n    await leaf.openFile(targetFile);\r\n    if (heading) {\r\n      let { editor } = leaf.view;\r\n      const pos = { line: heading.position.start.line, ch: 0 };\r\n      editor.setCursor(pos);\r\n      editor.scrollIntoView({ to: pos, from: pos }, true);\r\n    }\r\n  }\r\n\r\n  render_block_context(block) {\r\n    const block_headings = block.link.split(\".md\")[1].split(\"#\");\r\n    // starting with the last heading first, iterate through headings\r\n    let block_context = \"\";\r\n    for (let i = block_headings.length - 1; i >= 0; i--) {\r\n      if(block_context.length > 0) {\r\n        block_context = ` > ${block_context}`;\r\n      }\r\n      block_context = block_headings[i] + block_context;\r\n      // if block context is longer than N characters, break\r\n      if (block_context.length > 100) {\r\n        break;\r\n      }\r\n    }\r\n    // remove leading > if exists\r\n    if (block_context.startsWith(\" > \")) {\r\n      block_context = block_context.slice(3);\r\n    }\r\n    return block_context;\r\n\r\n  }\r\n\r\n  renderable_file_type(link) {\r\n    return (link.indexOf(\".md\") !== -1) && (link.indexOf(\".excalidraw\") === -1);\r\n  }\r\n\r\n  render_external_link_elm(meta){\r\n    if(meta.source) {\r\n      if(meta.source === \"Gmail\") meta.source = \"\uD83D\uDCE7 Gmail\";\r\n      return `<small>${meta.source}</small><br>${meta.title}`;\r\n    }\r\n    // remove http(s)://\r\n    let domain = meta.path.replace(/(^\\w+:|^)\\/\\//, \"\");\r\n    // separate domain from path\r\n    domain = domain.split(\"/\")[0];\r\n    // wrap domain in <small> and add line break\r\n    return `<small>\uD83C\uDF10 ${domain}</small><br>${meta.title}`;\r\n  }\r\n  // get all folders\r\n  async get_all_folders() {\r\n    if(!this.folders || this.folders.length === 0){\r\n      this.folders = await this.get_folders();\r\n    }\r\n    return this.folders;\r\n  }\r\n  // get folders, traverse non-hidden sub-folders\r\n  async get_folders(path = \"/\") {\r\n    let folders = (await this.app.vault.adapter.list(path)).folders;\r\n    let folder_list = [];\r\n    for (let i = 0; i < folders.length; i++) {\r\n      if (folders[i].startsWith(\".\")) continue;\r\n      folder_list.push(folders[i]);\r\n      folder_list = folder_list.concat(await this.get_folders(folders[i] + \"/\"));\r\n    }\r\n    return folder_list;\r\n  }\r\n\r\n\r\n  async sync_notes() {\r\n    // if license key is not set, return\r\n    if(!this.settings.license_key){\r\n      new Obsidian.Notice(\"Smart Connections: Supporter license key is required to sync notes to the ChatGPT Plugin server.\");\r\n      return;\r\n    }\r\n    console.log(\"syncing notes\");\r\n    // get all files in vault\r\n    const files = this.app.vault.getMarkdownFiles().filter((file) => {\r\n      // filter out file paths matching any strings in this.file_exclusions\r\n      for(let i = 0; i < this.file_exclusions.length; i++) {\r\n        if(file.path.indexOf(this.file_exclusions[i]) > -1) {\r\n          return false;\r\n        }\r\n      }\r\n      return true;\r\n    });\r\n    const notes = await this.build_notes_object(files);\r\n    console.log(\"object built\");\r\n    // save notes object to .smart-connections/notes.json\r\n    await this.app.vault.adapter.write(\".smart-connections/notes.json\", JSON.stringify(notes, null, 2));\r\n    console.log(\"notes saved\");\r\n    console.log(this.settings.license_key);\r\n    // POST notes object to server\r\n    const response = await (0, Obsidian.requestUrl)({\r\n      url: \"https://sync.smartconnections.app/sync\",\r\n      method: \"POST\",\r\n      headers: {\r\n        \"Content-Type\": \"application/json\",\r\n      },\r\n      contentType: \"application/json\",\r\n      body: JSON.stringify({\r\n        license_key: this.settings.license_key,\r\n        notes: notes\r\n      })\r\n    });\r\n    console.log(response);\r\n\r\n  }\r\n\r\n  async build_notes_object(files) {\r\n    let output = {};\r\n\r\n    for(let i = 0; i < files.length; i++) {\r\n      let file = files[i];\r\n      let parts = file.path.split(\"/\");\r\n      let current = output;\r\n\r\n      for (let ii = 0; ii < parts.length; ii++) {\r\n        let part = parts[ii];\r\n\r\n        if (ii === parts.length - 1) {\r\n          // This is a file\r\n          current[part] = await this.app.vault.cachedRead(file);\r\n        } else {\r\n          // This is a directory\r\n          if (!current[part]) {\r\n            current[part] = {};\r\n          }\r\n\r\n          current = current[part];\r\n        }\r\n      }\r\n    }\r\n\r\n    return output;\r\n  }\r\n\r\n}\r\n\r\nconst SMART_CONNECTIONS_VIEW_TYPE = \"smart-connections-view\";\r\nclass SmartConnectionsView extends Obsidian.ItemView {\r\n  constructor(leaf, plugin) {\r\n    super(leaf);\r\n    this.plugin = plugin;\r\n    this.nearest = null;\r\n    this.load_wait = null;\r\n  }\r\n  getViewType() {\r\n    return SMART_CONNECTIONS_VIEW_TYPE;\r\n  }\r\n\r\n  getDisplayText() {\r\n    return \"Smart Connections Files\";\r\n  }\r\n\r\n  getIcon() {\r\n    return \"smart-connections\";\r\n  }\r\n\r\n\r\n  set_message(message) {\r\n    const container = this.containerEl.children[1];\r\n    // clear container\r\n    container.empty();\r\n    // initiate top bar\r\n    this.initiate_top_bar(container);\r\n    // if mesage is an array, loop through and create a new p element for each message\r\n    if (Array.isArray(message)) {\r\n      for (let i = 0; i < message.length; i++) {\r\n        container.createEl(\"p\", { cls: \"sc_message\", text: message[i] });\r\n      }\r\n    }else{\r\n      // create p element with message\r\n      container.createEl(\"p\", { cls: \"sc_message\", text: message });\r\n    }\r\n  }\r\n  render_link_text(link, show_full_path=false) {\r\n    /**\r\n     * Begin internal links\r\n     */\r\n    // if show full path is false, remove file path\r\n    if (!show_full_path) {\r\n      link = link.split(\"/\").pop();\r\n    }\r\n    // if contains '#'\r\n    if (link.indexOf(\"#\") > -1) {\r\n      // split at .md\r\n      link = link.split(\".md\");\r\n      // wrap first part in <small> and add line break\r\n      link[0] = `<small>${link[0]}</small><br>`;\r\n      // join back together\r\n      link = link.join(\"\");\r\n      // replace '#' with ' \u00BB '\r\n      link = link.replace(/\\#/g, \" \u00BB \");\r\n    }else{\r\n      // remove '.md'\r\n      link = link.replace(\".md\", \"\");\r\n    }\r\n    return link;\r\n  }\r\n\r\n\r\n  set_nearest(nearest, nearest_context=null, results_only=false) {\r\n    // get container element\r\n    const container = this.containerEl.children[1];\r\n    // if results only is false, clear container and initiate top bar\r\n    if(!results_only){\r\n      // clear container\r\n      container.empty();\r\n      this.initiate_top_bar(container, nearest_context);\r\n    }\r\n    // update results\r\n    this.plugin.update_results(container, nearest);\r\n  }\r\n\r\n  initiate_top_bar(container, nearest_context=null) {\r\n    let top_bar;\r\n    // if top bar already exists, empty it\r\n    if ((container.children.length > 0) && (container.children[0].classList.contains(\"sc-top-bar\"))) {\r\n      top_bar = container.children[0];\r\n      top_bar.empty();\r\n    } else {\r\n      // init container for top bar\r\n      top_bar = container.createEl(\"div\", { cls: \"sc-top-bar\" });\r\n    }\r\n    // if highlighted text is not null, create p element with highlighted text\r\n    if (nearest_context) {\r\n      top_bar.createEl(\"p\", { cls: \"sc-context\", text: nearest_context });\r\n    }\r\n    // add chat button\r\n    const chat_button = top_bar.createEl(\"button\", { cls: \"sc-chat-button\" });\r\n    // add icon to chat button\r\n    Obsidian.setIcon(chat_button, \"message-square\");\r\n    // add click listener to chat button\r\n    chat_button.addEventListener(\"click\", () => {\r\n      // open chat\r\n      this.plugin.open_chat();\r\n    });\r\n    // add search button\r\n    const search_button = top_bar.createEl(\"button\", { cls: \"sc-search-button\" });\r\n    // add icon to search button\r\n    Obsidian.setIcon(search_button, \"search\");\r\n    // add click listener to search button\r\n    search_button.addEventListener(\"click\", () => {\r\n      // empty top bar\r\n      top_bar.empty();\r\n      // create input element\r\n      const search_container = top_bar.createEl(\"div\", { cls: \"search-input-container\" });\r\n      const input = search_container.createEl(\"input\", {\r\n        cls: \"sc-search-input\",\r\n        type: \"search\",\r\n        placeholder: \"Type to start search...\",\r\n      });\r\n      // focus input\r\n      input.focus();\r\n      // add keydown listener to input\r\n      input.addEventListener(\"keydown\", (event) => {\r\n        // if escape key is pressed\r\n        if (event.key === \"Escape\") {\r\n          this.clear_auto_searcher();\r\n          // clear top bar\r\n          this.initiate_top_bar(container, nearest_context);\r\n        }\r\n      });\r\n\r\n      // add keyup listener to input\r\n      input.addEventListener(\"keyup\", (event) => {\r\n        // if this.search_timeout is not null then clear it and set to null\r\n        this.clear_auto_searcher();\r\n        // get search term\r\n        const search_term = input.value;\r\n        // if enter key is pressed\r\n        if (event.key === \"Enter\" && search_term !== \"\") {\r\n          this.search(search_term);\r\n        }\r\n        // if any other key is pressed and input is not empty then wait 500ms and make_connections\r\n        else if (search_term !== \"\") {\r\n          // clear timeout\r\n          clearTimeout(this.search_timeout);\r\n          // set timeout\r\n          this.search_timeout = setTimeout(() => {\r\n            this.search(search_term, true);\r\n          }, 700);\r\n        }\r\n      });\r\n    });\r\n  }\r\n\r\n  // render buttons: \"create\" and \"retry\" for loading embeddings.json file\r\n  render_embeddings_buttons() {\r\n    // get container element\r\n    const container = this.containerEl.children[1];\r\n    // clear container\r\n    container.empty();\r\n    // create heading that says \"Embeddings file not found\"\r\n    container.createEl(\"h2\", { cls: \"scHeading\", text: \"Embeddings file not found\" });\r\n    // create div for buttons\r\n    const button_div = container.createEl(\"div\", { cls: \"scButtonDiv\" });\r\n    // create \"create\" button\r\n    const create_button = button_div.createEl(\"button\", { cls: \"scButton\", text: \"Create embeddings.json\" });\r\n    // note that creating embeddings.json file will trigger bulk embedding and may take a while\r\n    button_div.createEl(\"p\", { cls: \"scButtonNote\", text: \"Warning: Creating embeddings.json file will trigger bulk embedding and may take a while\" });\r\n    // create \"retry\" button\r\n    const retry_button = button_div.createEl(\"button\", { cls: \"scButton\", text: \"Retry\" });\r\n    // try to load embeddings.json file again\r\n    button_div.createEl(\"p\", { cls: \"scButtonNote\", text: \"If embeddings.json file already exists, click 'Retry' to load it\" });\r\n\r\n    // add click event to \"create\" button\r\n    create_button.addEventListener(\"click\", async (event) => {\r\n      // create embeddings.json file\r\n      await this.plugin.smart_vec_lite.init_embeddings_file();\r\n      // reload view\r\n      await this.render_connections();\r\n    });\r\n\r\n    // add click event to \"retry\" button\r\n    retry_button.addEventListener(\"click\", async (event) => {\r\n      console.log(\"retrying to load embeddings.json file\");\r\n      // reload embeddings.json file\r\n      await this.plugin.init_vecs();\r\n      // reload view\r\n      await this.render_connections();\r\n    });\r\n  }\r\n\r\n  async onOpen() {\r\n    const container = this.containerEl.children[1];\r\n    container.empty();\r\n    // placeholder text\r\n    container.createEl(\"p\", { cls: \"scPlaceholder\", text: \"Open a note to find connections.\" });\r\n\r\n    // runs when file is opened\r\n    this.plugin.registerEvent(this.app.workspace.on('file-open', (file) => {\r\n      // if no file is open, return\r\n      if(!file) {\r\n        // console.log(\"no file open, returning\");\r\n        return;\r\n      }\r\n      // return if file type is not supported\r\n      if(SUPPORTED_FILE_TYPES.indexOf(file.extension) === -1) {\r\n        return this.set_message([\r\n          \"File: \"+file.name\r\n          ,\"Unsupported file type (Supported: \"+SUPPORTED_FILE_TYPES.join(\", \")+\")\"\r\n        ]);\r\n      }\r\n      // run render_connections after 1 second to allow for file to load\r\n      if(this.load_wait){\r\n        clearTimeout(this.load_wait);\r\n      }\r\n      this.load_wait = setTimeout(() => {\r\n        this.render_connections(file);\r\n        this.load_wait = null;\r\n      }, 1000);\r\n\r\n    }));\r\n\r\n    this.app.workspace.registerHoverLinkSource(SMART_CONNECTIONS_VIEW_TYPE, {\r\n        display: 'Smart Connections Files',\r\n        defaultMod: true,\r\n    });\r\n    this.app.workspace.registerHoverLinkSource(SMART_CONNECTIONS_CHAT_VIEW_TYPE, {\r\n        display: 'Smart Chat Links',\r\n        defaultMod: true,\r\n    });\r\n\r\n    this.app.workspace.onLayoutReady(this.initialize.bind(this));\r\n\r\n  }\r\n\r\n  async initialize() {\r\n    this.set_message(\"Loading embeddings file...\");\r\n    const vecs_intiated = await this.plugin.init_vecs();\r\n    if(vecs_intiated){\r\n      this.set_message(\"Embeddings file loaded.\");\r\n      await this.render_connections();\r\n    }else{\r\n      this.render_embeddings_buttons();\r\n    }\r\n\r\n    /**\r\n     * EXPERIMENTAL\r\n     * - window-based API access\r\n     * - code-block rendering\r\n     */\r\n    this.api = new SmartConnectionsViewApi(this.app, this.plugin, this);\r\n    // register API to global window object\r\n    (window[\"SmartConnectionsViewApi\"] = this.api) && this.register(() => delete window[\"SmartConnectionsViewApi\"]);\r\n\r\n  }\r\n\r\n  async onClose() {\r\n    console.log(\"closing smart connections view\");\r\n    this.app.workspace.unregisterHoverLinkSource(SMART_CONNECTIONS_VIEW_TYPE);\r\n    this.plugin.view = null;\r\n  }\r\n\r\n  async render_connections(context=null) {\r\n    console.log(\"rendering connections\");\r\n    // if API key is not set then update view message\r\n    if(!this.plugin.settings.api_key) {\r\n      this.set_message(\"An OpenAI API key is required to make Smart Connections\");\r\n      return;\r\n    }\r\n    if(!this.plugin.embeddings_loaded){\r\n      await this.plugin.init_vecs();\r\n    }\r\n    // if embedding still not loaded, return\r\n    if(!this.plugin.embeddings_loaded) {\r\n      console.log(\"embeddings files still not loaded or yet to be created\");\r\n      this.render_embeddings_buttons();\r\n      return;\r\n    }\r\n    this.set_message(\"Making Smart Connections...\");\r\n    /**\r\n     * Begin highlighted-text-level search\r\n     */\r\n    if(typeof context === \"string\") {\r\n      const highlighted_text = context;\r\n      // get embedding for highlighted text\r\n      await this.search(highlighted_text);\r\n      return; // ends here if context is a string\r\n    }\r\n\r\n    /**\r\n     * Begin file-level search\r\n     */\r\n    this.nearest = null;\r\n    this.interval_count = 0;\r\n    this.rendering = false;\r\n    this.file = context;\r\n    // if this.interval is set then clear it\r\n    if(this.interval) {\r\n      clearInterval(this.interval);\r\n      this.interval = null;\r\n    }\r\n    // set interval to check if nearest is set\r\n    this.interval = setInterval(() => {\r\n      if(!this.rendering){\r\n        if(this.file instanceof Obsidian.TFile) {\r\n          this.rendering = true;\r\n          this.render_note_connections(this.file);\r\n        }else{\r\n          // get current note\r\n          this.file = this.app.workspace.getActiveFile();\r\n          // if still no current note then return\r\n          if(!this.file && this.count > 1) {\r\n            clearInterval(this.interval);\r\n            this.set_message(\"No active file\");\r\n            return;\r\n          }\r\n        }\r\n      }else{\r\n        if(this.nearest) {\r\n          clearInterval(this.interval);\r\n          // if nearest is a string then update view message\r\n          if (typeof this.nearest === \"string\") {\r\n            this.set_message(this.nearest);\r\n          } else {\r\n            // set nearest connections\r\n            this.set_nearest(this.nearest, \"File: \" + this.file.name);\r\n          }\r\n          // if render_log.failed_embeddings then update failed_embeddings.txt\r\n          if (this.plugin.render_log.failed_embeddings.length > 0) {\r\n            this.plugin.save_failed_embeddings();\r\n          }\r\n          // get object keys of render_log\r\n          this.plugin.output_render_log();\r\n          return;\r\n        }else{\r\n          this.interval_count++;\r\n          this.set_message(\"Making Smart Connections...\"+this.interval_count);\r\n        }\r\n      }\r\n    }, 10);\r\n  }\r\n\r\n  async render_note_connections(file) {\r\n    this.nearest = await this.plugin.find_note_connections(file);\r\n  }\r\n\r\n  clear_auto_searcher() {\r\n    if (this.search_timeout) {\r\n      clearTimeout(this.search_timeout);\r\n      this.search_timeout = null;\r\n    }\r\n  }\r\n\r\n  async search(search_text, results_only=false) {\r\n    const nearest = await this.plugin.api.search(search_text);\r\n    // render results in view with first 100 characters of search text\r\n    const nearest_context = `Selection: \"${search_text.length > 100 ? search_text.substring(0, 100) + \"...\" : search_text}\"`;\r\n    this.set_nearest(nearest, nearest_context, results_only);\r\n  }\r\n\r\n}\r\nclass SmartConnectionsViewApi {\r\n  constructor(app, plugin, view) {\r\n    this.app = app;\r\n    this.plugin = plugin;\r\n    this.view = view;\r\n  }\r\n  async search(search_text) {\r\n    return await this.plugin.api.search(search_text);\r\n  }\r\n  // trigger reload of embeddings file\r\n  async reload_embeddings_file() {\r\n    await this.plugin.init_vecs();\r\n    await this.view.render_connections();\r\n  }\r\n  async init_vecs() {\r\n    this.smart_vec_lite = new VecLite({\r\n      folder_path: \".smart-connections\",\r\n      exists_adapter: this.app.vault.adapter.exists.bind(\r\n        this.app.vault.adapter\r\n      ),\r\n      mkdir_adapter: this.app.vault.adapter.mkdir.bind(this.app.vault.adapter),\r\n      read_adapter: this.app.vault.adapter.read.bind(this.app.vault.adapter),\r\n      rename_adapter: this.app.vault.adapter.rename.bind(\r\n        this.app.vault.adapter\r\n      ),\r\n      stat_adapter: this.app.vault.adapter.stat.bind(this.app.vault.adapter),\r\n      write_adapter: this.app.vault.adapter.write.bind(this.app.vault.adapter),\r\n    });\r\n    this.embeddings_loaded = await this.smart_vec_lite.load();\r\n    return this.embeddings_loaded;\r\n  }\r\n}\r\nclass ScSearchApi {\r\n  constructor(app, plugin) {\r\n    this.app = app;\r\n    this.plugin = plugin;\r\n  }\r\n  async search (search_text, filter={}) {\r\n    filter = {\r\n      skip_sections: this.plugin.settings.skip_sections,\r\n      ...filter\r\n    }\r\n    let nearest = [];\r\n    const resp = await this.plugin.request_embedding_from_input(search_text);\r\n    if (resp && resp.data && resp.data[0] && resp.data[0].embedding) {\r\n      nearest = this.plugin.smart_vec_lite.nearest(resp.data[0].embedding, filter);\r\n    } else {\r\n      // resp is null, undefined, or missing data\r\n      new Obsidian.Notice(\"Smart Connections: Error getting embedding\");\r\n    }\r\n    return nearest;\r\n  }\r\n}\r\n\r\nclass SmartConnectionsSettingsTab extends Obsidian.PluginSettingTab {\r\n  constructor(app, plugin) {\r\n    super(app, plugin);\r\n    this.plugin = plugin;\r\n  }\r\n  display() {\r\n    const {\r\n      containerEl\r\n    } = this;\r\n    containerEl.empty();\r\n    containerEl.createEl(\"h2\", {\r\n      text: \"Supporter Settings\"\r\n    });\r\n    // list supporter benefits\r\n    containerEl.createEl(\"p\", {\r\n      text: \"As a Smart Connections \\\"Supporter\\\", fast-track your PKM journey with priority perks and pioneering innovations.\"\r\n    });\r\n    // three list items\r\n    const supporter_benefits_list = containerEl.createEl(\"ul\");\r\n    supporter_benefits_list.createEl(\"li\", {\r\n      text: \"Enjoy swift, top-priority support.\"\r\n    });\r\n    supporter_benefits_list.createEl(\"li\", {\r\n      text: \"Gain early access to version 2 (includes local embedding model).\"\r\n    });\r\n    supporter_benefits_list.createEl(\"li\", {\r\n      text: \"Stay informed and engaged with exclusive supporter-only communications.\"\r\n    });\r\n    // add a text input to enter supporter license key\r\n    new Obsidian.Setting(containerEl).setName(\"Supporter License Key\").setDesc(\"Note: this is not required to use Smart Connections.\").addText((text) => text.setPlaceholder(\"Enter your license_key\").setValue(this.plugin.settings.license_key).onChange(async (value) => {\r\n      this.plugin.settings.license_key = value.trim();\r\n      await this.plugin.saveSettings(true);\r\n    }));\r\n    // button \"get v2\"\r\n    new Obsidian.Setting(containerEl).setName(\"Get v2\").setDesc(\"Get v2 (warning: very early beta release, likely to crash, please send issues directly to the supporter email for quick response)\").addButton((button) => button.setButtonText(\"Get v2 (unstable)\").onClick(async () => {\r\n      await this.plugin.update_to_v2();\r\n    }));\r\n    // add button to trigger sync notes to use with ChatGPT\r\n    new Obsidian.Setting(containerEl).setName(\"Sync Notes\").setDesc(\"Make notes available via the Smart Connections ChatGPT Plugin. Respects exclusion settings configured below.\").addButton((button) => button.setButtonText(\"Sync Notes\").onClick(async () => {\r\n      // sync notes\r\n      await this.plugin.sync_notes();\r\n    }));\r\n    // add button to become a supporter\r\n    new Obsidian.Setting(containerEl).setName(\"Become a Supporter\").setDesc(\"Become a Supporter\").addButton((button) => button.setButtonText(\"Become a Supporter\").onClick(async () => {\r\n      const payment_pages = [\r\n        \"https://buy.stripe.com/9AQ5kO5QnbAWgGAbIY\",\r\n        \"https://buy.stripe.com/9AQ7sWemT48u1LGcN4\"\r\n      ];\r\n      if(!this.plugin.payment_page_index){\r\n        this.plugin.payment_page_index = Math.round(Math.random());\r\n      }\r\n      // open supporter page in browser\r\n      window.open(payment_pages[this.plugin.payment_page_index]);\r\n    }));\r\n\r\n\r\n    containerEl.createEl(\"h2\", {\r\n      text: \"OpenAI Settings\"\r\n    });\r\n    // add a text input to enter the API key\r\n    new Obsidian.Setting(containerEl).setName(\"OpenAI API Key\").setDesc(\"Required: an OpenAI API key is currently required to use Smart Connections.\").addText((text) => text.setPlaceholder(\"Enter your api_key\").setValue(this.plugin.settings.api_key).onChange(async (value) => {\r\n      this.plugin.settings.api_key = value.trim();\r\n      await this.plugin.saveSettings(true);\r\n    }));\r\n    // add a button to test the API key is working\r\n    new Obsidian.Setting(containerEl).setName(\"Test API Key\").setDesc(\"Test API Key\").addButton((button) => button.setButtonText(\"Test API Key\").onClick(async () => {\r\n      // test API key\r\n      const resp = await this.plugin.test_api_key();\r\n      if(resp) {\r\n        new Obsidian.Notice(\"Smart Connections: API key is valid\");\r\n      }else{\r\n        new Obsidian.Notice(\"Smart Connections: API key is not working as expected!\");\r\n      }\r\n    }));\r\n    // add dropdown to select the model\r\n    new Obsidian.Setting(containerEl).setName(\"Smart Chat Model\").setDesc(\"Select a model to use with Smart Chat.\").addDropdown((dropdown) => {\r\n      dropdown.addOption(\"gpt-3.5-turbo-16k\", \"gpt-3.5-turbo-16k\");\r\n      dropdown.addOption(\"gpt-4\", \"gpt-4 (limited access, 8k)\");\r\n      dropdown.addOption(\"gpt-3.5-turbo\", \"gpt-3.5-turbo (4k)\");\r\n      dropdown.addOption(\"gpt-4-1106-preview\", \"gpt-4-turbo (128k)\");\r\n      dropdown.onChange(async (value) => {\r\n        this.plugin.settings.smart_chat_model = value;\r\n        await this.plugin.saveSettings();\r\n      });\r\n      dropdown.setValue(this.plugin.settings.smart_chat_model);\r\n    });\r\n    // language\r\n    new Obsidian.Setting(containerEl).setName(\"Default Language\").setDesc(\"Default language to use for Smart Chat. Changes which self-referential pronouns will trigger lookup of your notes.\").addDropdown((dropdown) => {\r\n      // get Object keys from pronous\r\n      const languages = Object.keys(SMART_TRANSLATION);\r\n      for(let i = 0; i < languages.length; i++) {\r\n        dropdown.addOption(languages[i], languages[i]);\r\n      }\r\n      dropdown.onChange(async (value) => {\r\n        this.plugin.settings.language = value;\r\n        await this.plugin.saveSettings();\r\n        self_ref_pronouns_list.setText(this.get_self_ref_list());\r\n        // if chat view is open then run new_chat()\r\n        const chat_view = this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE).length > 0 ? this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE)[0].view : null;\r\n        if(chat_view) {\r\n          chat_view.new_chat();\r\n        }\r\n      });\r\n      dropdown.setValue(this.plugin.settings.language);\r\n    });\r\n    // list current self-referential pronouns\r\n    const self_ref_pronouns_list = containerEl.createEl(\"span\", {\r\n      text: this.get_self_ref_list()\r\n    });\r\n    new Obsidian.Setting(containerEl).setName(\"Cut off frontmatter\").setDesc(\"Cut off frontmatter in the prompt to gain characters in reply generation\").addToggle((toggle) => { toggle.setValue(this.plugin.settings.cut_off_frontmatter).onChange(async (value) => { this.plugin.settings.cut_off_frontmatter = value; await this.plugin.saveSettings(); }); });\r\n    containerEl.createEl(\"h2\", {\r\n      text: \"Exclusions\"\r\n    });\r\n    // list file exclusions\r\n    new Obsidian.Setting(containerEl).setName(\"file_exclusions\").setDesc(\"'Excluded file' matchers separated by a comma.\").addText((text) => text.setPlaceholder(\"drawings,prompts/logs\").setValue(this.plugin.settings.file_exclusions).onChange(async (value) => {\r\n      this.plugin.settings.file_exclusions = value;\r\n      await this.plugin.saveSettings();\r\n    }));\r\n    // list folder exclusions\r\n    new Obsidian.Setting(containerEl).setName(\"folder_exclusions\").setDesc(\"'Excluded folder' matchers separated by a comma.\").addText((text) => text.setPlaceholder(\"drawings,prompts/logs\").setValue(this.plugin.settings.folder_exclusions).onChange(async (value) => {\r\n      this.plugin.settings.folder_exclusions = value;\r\n      await this.plugin.saveSettings();\r\n    }));\r\n    // list path only matchers\r\n    new Obsidian.Setting(containerEl).setName(\"path_only\").setDesc(\"'Path only' matchers separated by a comma.\").addText((text) => text.setPlaceholder(\"drawings,prompts/logs\").setValue(this.plugin.settings.path_only).onChange(async (value) => {\r\n      this.plugin.settings.path_only = value;\r\n      await this.plugin.saveSettings();\r\n    }));\r\n    // list header exclusions\r\n    new Obsidian.Setting(containerEl).setName(\"header_exclusions\").setDesc(\"'Excluded header' matchers separated by a comma. Works for 'blocks' only.\").addText((text) => text.setPlaceholder(\"drawings,prompts/logs\").setValue(this.plugin.settings.header_exclusions).onChange(async (value) => {\r\n      this.plugin.settings.header_exclusions = value;\r\n      await this.plugin.saveSettings();\r\n    }));\r\n    containerEl.createEl(\"h2\", {\r\n      text: \"Display\"\r\n    });\r\n    // toggle showing full path in view\r\n    new Obsidian.Setting(containerEl).setName(\"show_full_path\").setDesc(\"Show full path in view.\").addToggle((toggle) => toggle.setValue(this.plugin.settings.show_full_path).onChange(async (value) => {\r\n      this.plugin.settings.show_full_path = value;\r\n      await this.plugin.saveSettings(true);\r\n    }));\r\n    // toggle expanded view by default\r\n    new Obsidian.Setting(containerEl).setName(\"expanded_view\").setDesc(\"Expanded view by default.\").addToggle((toggle) => toggle.setValue(this.plugin.settings.expanded_view).onChange(async (value) => {\r\n      this.plugin.settings.expanded_view = value;\r\n      await this.plugin.saveSettings(true);\r\n    }));\r\n    // toggle group nearest by file\r\n    new Obsidian.Setting(containerEl).setName(\"group_nearest_by_file\").setDesc(\"Group nearest by file.\").addToggle((toggle) => toggle.setValue(this.plugin.settings.group_nearest_by_file).onChange(async (value) => {\r\n      this.plugin.settings.group_nearest_by_file = value;\r\n      await this.plugin.saveSettings(true);\r\n    }));\r\n    // toggle view_open on Obsidian startup\r\n    new Obsidian.Setting(containerEl).setName(\"view_open\").setDesc(\"Open view on Obsidian startup.\").addToggle((toggle) => toggle.setValue(this.plugin.settings.view_open).onChange(async (value) => {\r\n      this.plugin.settings.view_open = value;\r\n      await this.plugin.saveSettings(true);\r\n    }));\r\n    // toggle chat_open on Obsidian startup\r\n    new Obsidian.Setting(containerEl).setName(\"chat_open\").setDesc(\"Open view on Obsidian startup.\").addToggle((toggle) => toggle.setValue(this.plugin.settings.chat_open).onChange(async (value) => {\r\n      this.plugin.settings.chat_open = value;\r\n      await this.plugin.saveSettings(true);\r\n    }));\r\n    // open in big view or small view\r\n    new Obsidian.Setting(containerEl).setName(\"open_in_big_view\").setDesc(\"Open in big view or small view.\").addDropdown((dropdown) => {\r\n      dropdown.addOption(false, \"Right pane (small)\");\r\n      dropdown.addOption(true, \"Main pane (big)\");\r\n      dropdown.setValue(this.plugin.settings.open_in_big_view);\r\n      dropdown.onChange(async (value) => {\r\n        this.plugin.settings.open_in_big_view = JSON.parse(value);\r\n        await this.plugin.saveSettings(true);\r\n        this.plugin.open_chat();\r\n\r\n      });\r\n    });\r\n    containerEl.createEl(\"h2\", {\r\n      text: \"Advanced\"\r\n    });\r\n    // toggle log_render\r\n    new Obsidian.Setting(containerEl).setName(\"log_render\").setDesc(\"Log render details to console (includes token_usage).\").addToggle((toggle) => toggle.setValue(this.plugin.settings.log_render).onChange(async (value) => {\r\n      this.plugin.settings.log_render = value;\r\n      await this.plugin.saveSettings(true);\r\n    }));\r\n    // toggle files in log_render\r\n    new Obsidian.Setting(containerEl).setName(\"log_render_files\").setDesc(\"Log embedded objects paths with log render (for debugging).\").addToggle((toggle) => toggle.setValue(this.plugin.settings.log_render_files).onChange(async (value) => {\r\n      this.plugin.settings.log_render_files = value;\r\n      await this.plugin.saveSettings(true);\r\n    }));\r\n    // toggle skip_sections\r\n    new Obsidian.Setting(containerEl).setName(\"skip_sections\").setDesc(\"Skips making connections to specific sections within notes. Warning: reduces usefulness for large files and requires 'Force Refresh' for sections to work in the future.\").addToggle((toggle) => toggle.setValue(this.plugin.settings.skip_sections).onChange(async (value) => {\r\n      this.plugin.settings.skip_sections = value;\r\n      await this.plugin.saveSettings(true);\r\n    }));\r\n    // test file writing by creating a test file, then writing additional data to the file, and returning any error text if it fails\r\n    containerEl.createEl(\"h3\", {\r\n      text: \"Test File Writing\"\r\n    });\r\n    // manual save button\r\n    containerEl.createEl(\"h3\", {\r\n      text: \"Manual Save\"\r\n    });\r\n    let manual_save_results = containerEl.createEl(\"div\");\r\n    new Obsidian.Setting(containerEl).setName(\"manual_save\").setDesc(\"Save current embeddings\").addButton((button) => button.setButtonText(\"Manual Save\").onClick(async () => {\r\n      // confirm\r\n      if (confirm(\"Are you sure you want to save your current embeddings?\")) {\r\n        // save\r\n        try{\r\n          await this.plugin.save_embeddings_to_file(true);\r\n          manual_save_results.innerHTML = \"Embeddings saved successfully.\";\r\n        }catch(e){\r\n          manual_save_results.innerHTML = \"Embeddings failed to save. Error: \" + e;\r\n        }\r\n      }\r\n    }));\r\n\r\n    // list previously failed files\r\n    containerEl.createEl(\"h3\", {\r\n      text: \"Previously failed files\"\r\n    });\r\n    let failed_list = containerEl.createEl(\"div\");\r\n    this.draw_failed_files_list(failed_list);\r\n\r\n    // force refresh button\r\n    containerEl.createEl(\"h3\", {\r\n      text: \"Force Refresh\"\r\n    });\r\n    new Obsidian.Setting(containerEl).setName(\"force_refresh\").setDesc(\"WARNING: DO NOT use unless you know what you are doing! This will delete all of your current embeddings from OpenAI and trigger reprocessing of your entire vault!\").addButton((button) => button.setButtonText(\"Force Refresh\").onClick(async () => {\r\n      // confirm\r\n      if (confirm(\"Are you sure you want to Force Refresh? By clicking yes you confirm that you understand the consequences of this action.\")) {\r\n        // force refresh\r\n        await this.plugin.force_refresh_embeddings_file();\r\n      }\r\n    }));\r\n\r\n  }\r\n  get_self_ref_list() {\r\n    return \"Current: \" + SMART_TRANSLATION[this.plugin.settings.language].pronous.join(\", \");\r\n  }\r\n\r\n  draw_failed_files_list(failed_list) {\r\n    failed_list.empty();\r\n    if(this.plugin.settings.failed_files.length > 0) {\r\n      // add message that these files will be skipped until manually retried\r\n      failed_list.createEl(\"p\", {\r\n        text: \"The following files failed to process and will be skipped until manually retried.\"\r\n      });\r\n      let list = failed_list.createEl(\"ul\");\r\n      for (let failed_file of this.plugin.settings.failed_files) {\r\n        list.createEl(\"li\", {\r\n          text: failed_file\r\n        });\r\n      }\r\n      // add button to retry failed files only\r\n      new Obsidian.Setting(failed_list).setName(\"retry_failed_files\").setDesc(\"Retry failed files only\").addButton((button) => button.setButtonText(\"Retry failed files only\").onClick(async () => {\r\n        // clear failed_list element\r\n        failed_list.empty();\r\n        // set \"retrying\" text\r\n        failed_list.createEl(\"p\", {\r\n          text: \"Retrying failed files...\"\r\n        });\r\n        await this.plugin.retry_failed_files();\r\n        // redraw failed files list\r\n        this.draw_failed_files_list(failed_list);\r\n      }));\r\n    }else{\r\n      failed_list.createEl(\"p\", {\r\n        text: \"No failed files\"\r\n      });\r\n    }\r\n  }\r\n}\r\n\r\nfunction line_is_heading(line) {\r\n  return (line.indexOf(\"#\") === 0) && (['#', ' '].indexOf(line[1]) !== -1);\r\n}\r\n\r\nconst SMART_CONNECTIONS_CHAT_VIEW_TYPE = \"smart-connections-chat-view\";\r\n\r\nclass SmartConnectionsChatView extends Obsidian.ItemView {\r\n  constructor(leaf, plugin) {\r\n    super(leaf);\r\n    this.plugin = plugin;\r\n    this.active_elm = null;\r\n    this.active_stream = null;\r\n    this.brackets_ct = 0;\r\n    this.chat = null;\r\n    this.chat_box = null;\r\n    this.chat_container = null;\r\n    this.current_chat_ml = [];\r\n    this.files = [];\r\n    this.last_from = null;\r\n    this.message_container = null;\r\n    this.prevent_input = false;\r\n  }\r\n  getDisplayText() {\r\n    return \"Smart Connections Chat\";\r\n  }\r\n  getIcon() {\r\n    return \"message-square\";\r\n  }\r\n  getViewType() {\r\n    return SMART_CONNECTIONS_CHAT_VIEW_TYPE;\r\n  }\r\n  onOpen() {\r\n    this.new_chat();\r\n    this.plugin.get_all_folders(); // sets this.plugin.folders necessary for folder-context\r\n  }\r\n  onClose() {\r\n    this.chat.save_chat();\r\n    this.app.workspace.unregisterHoverLinkSource(SMART_CONNECTIONS_CHAT_VIEW_TYPE);\r\n  }\r\n  render_chat() {\r\n    this.containerEl.empty();\r\n    this.chat_container = this.containerEl.createDiv(\"sc-chat-container\");\r\n    // render plus sign for clear button\r\n    this.render_top_bar();\r\n    // render chat messages container\r\n    this.render_chat_box();\r\n    // render chat input\r\n    this.render_chat_input();\r\n    this.plugin.render_brand(this.containerEl, \"chat\");\r\n  }\r\n  // render plus sign for clear button\r\n  render_top_bar() {\r\n    // create container for clear button\r\n    let top_bar_container = this.chat_container.createDiv(\"sc-top-bar-container\");\r\n    // render the name of the chat in an input box (pop content after last hyphen in chat_id)\r\n    let chat_name =this.chat.name();\r\n    let chat_name_input = top_bar_container.createEl(\"input\", {\r\n      attr: {\r\n        type: \"text\",\r\n        value: chat_name\r\n      },\r\n      cls: \"sc-chat-name-input\"\r\n    });\r\n    chat_name_input.addEventListener(\"change\", this.rename_chat.bind(this));\r\n\r\n    // create button to Smart View\r\n    let smart_view_btn = this.create_top_bar_button(top_bar_container, \"Smart View\", \"smart-connections\");\r\n    smart_view_btn.addEventListener(\"click\", this.open_smart_view.bind(this));\r\n    // create button to save chat\r\n    let save_btn = this.create_top_bar_button(top_bar_container, \"Save Chat\", \"save\");\r\n    save_btn.addEventListener(\"click\", this.save_chat.bind(this));\r\n    // create button to open chat history modal\r\n    let history_btn = this.create_top_bar_button(top_bar_container, \"Chat History\", \"history\");\r\n    history_btn.addEventListener(\"click\", this.open_chat_history.bind(this));\r\n    // create button to start new chat\r\n    const new_chat_btn = this.create_top_bar_button(top_bar_container, \"New Chat\", \"plus\");\r\n    new_chat_btn.addEventListener(\"click\", this.new_chat.bind(this));\r\n  }\r\n  async open_chat_history() {\r\n    const folder = await this.app.vault.adapter.list(\".smart-connections/chats\");\r\n    this.files = folder.files.map((file) => {\r\n      return file.replace(\".smart-connections/chats/\", \"\").replace(\".json\", \"\");\r\n    });\r\n    // open chat history modal\r\n    if (!this.modal)\r\n      this.modal = new SmartConnectionsChatHistoryModal(this.app, this);\r\n    this.modal.open();\r\n  }\r\n\r\n  create_top_bar_button(top_bar_container, title, icon=null) {\r\n    let btn = top_bar_container.createEl(\"button\", {\r\n      attr: {\r\n        title: title\r\n      }\r\n    });\r\n    if(icon){\r\n      Obsidian.setIcon(btn, icon);\r\n    }else{\r\n      btn.innerHTML = title;\r\n    }\r\n    return btn;\r\n  }\r\n  // render new chat\r\n  new_chat() {\r\n    this.clear_chat();\r\n    this.render_chat();\r\n    // render initial message from assistant (don't use render_message to skip adding to chat history)\r\n    this.new_messsage_bubble(\"assistant\");\r\n    this.active_elm.innerHTML = '<p>' + SMART_TRANSLATION[this.plugin.settings.language].initial_message+'</p>';\r\n  }\r\n  // open a chat from the chat history modal\r\n  async open_chat(chat_id) {\r\n    this.clear_chat();\r\n    await this.chat.load_chat(chat_id);\r\n    this.render_chat();\r\n    for (let i = 0; i < this.chat.chat_ml.length; i++) {\r\n      await this.render_message(this.chat.chat_ml[i].content, this.chat.chat_ml[i].role);\r\n    }\r\n  }\r\n  // clear current chat state\r\n  clear_chat() {\r\n    if (this.chat) {\r\n      this.chat.save_chat();\r\n    }\r\n    this.chat = new SmartConnectionsChatModel(this.plugin);\r\n    // if this.dotdotdot_interval is not null, clear interval\r\n    if (this.dotdotdot_interval) {\r\n      clearInterval(this.dotdotdot_interval);\r\n    }\r\n    // clear current chat ml\r\n    this.current_chat_ml = [];\r\n    // update prevent input\r\n    this.end_stream();\r\n  }\r\n\r\n  rename_chat(event) {\r\n    let new_chat_name = event.target.value;\r\n    this.chat.rename_chat(new_chat_name);\r\n  }\r\n\r\n  // save current chat\r\n  save_chat() {\r\n    this.chat.save_chat();\r\n    new Obsidian.Notice(\"[Smart Connections] Chat saved\");\r\n  }\r\n\r\n  open_smart_view() {\r\n    this.plugin.open_view();\r\n  }\r\n  // render chat messages container\r\n  render_chat_box() {\r\n    // create container for chat messages\r\n    this.chat_box = this.chat_container.createDiv(\"sc-chat-box\");\r\n    // create container for message\r\n    this.message_container = this.chat_box.createDiv(\"sc-message-container\");\r\n  }\r\n  // open file suggestion modal\r\n  open_file_suggestion_modal() {\r\n    // open file suggestion modal\r\n    if(!this.file_selector) this.file_selector = new SmartConnectionsFileSelectModal(this.app, this);\r\n    this.file_selector.open();\r\n  }\r\n  // open folder suggestion modal\r\n  async open_folder_suggestion_modal() {\r\n    // open folder suggestion modal\r\n    if(!this.folder_selector){\r\n      this.folder_selector = new SmartConnectionsFolderSelectModal(this.app, this);\r\n    }\r\n    this.folder_selector.open();\r\n  }\r\n  // insert_selection from file suggestion modal\r\n  insert_selection(insert_text) {\r\n    // get caret position\r\n    let caret_pos = this.textarea.selectionStart;\r\n    // get text before caret\r\n    let text_before = this.textarea.value.substring(0, caret_pos);\r\n    // get text after caret\r\n    let text_after = this.textarea.value.substring(caret_pos, this.textarea.value.length);\r\n    // insert text\r\n    this.textarea.value = text_before + insert_text + text_after;\r\n    // set caret position\r\n    this.textarea.selectionStart = caret_pos + insert_text.length;\r\n    this.textarea.selectionEnd = caret_pos + insert_text.length;\r\n    // focus on textarea\r\n    this.textarea.focus();\r\n  }\r\n\r\n  // render chat textarea and button\r\n  render_chat_input() {\r\n    // create container for chat input\r\n    let chat_input = this.chat_container.createDiv(\"sc-chat-form\");\r\n    // create textarea\r\n    this.textarea = chat_input.createEl(\"textarea\", {\r\n      cls: \"sc-chat-input\",\r\n      attr: {\r\n        placeholder: SMART_TRANSLATION[this.plugin.settings.language].try_placeholder\r\n      }\r\n    });\r\n    // use contenteditable instead of textarea\r\n    // this.textarea = chat_input.createEl(\"div\", {cls: \"sc-chat-input\", attr: {contenteditable: true}});\r\n    // add event listener to listen for shift+enter\r\n    chat_input.addEventListener(\"keyup\", (e) => {\r\n      if([\"[\", \"/\"].indexOf(e.key) === -1) return; // skip if key is not [ or /\r\n      const caret_pos = this.textarea.selectionStart;\r\n      // if key is open square bracket\r\n      if (e.key === \"[\") {\r\n        // if previous char is [\r\n        if(this.textarea.value[caret_pos - 2] === \"[\"){\r\n          // open file suggestion modal\r\n          this.open_file_suggestion_modal();\r\n          return;\r\n        }\r\n      }else{\r\n        this.brackets_ct = 0;\r\n      }\r\n      // if / is pressed\r\n      if (e.key === \"/\") {\r\n        // get caret position\r\n        // if this is first char or previous char is space\r\n        if (this.textarea.value.length === 1 || this.textarea.value[caret_pos - 2] === \" \") {\r\n          // open folder suggestion modal\r\n          this.open_folder_suggestion_modal();\r\n          return;\r\n        }\r\n      }\r\n\r\n    });\r\n\r\n    chat_input.addEventListener(\"keydown\", (e) => {\r\n      if (e.key === \"Enter\" && e.shiftKey) {\r\n        e.preventDefault();\r\n        if(this.prevent_input){\r\n          console.log(\"wait until current response is finished\");\r\n          new Obsidian.Notice(\"[Smart Connections] Wait until current response is finished\");\r\n          return;\r\n        }\r\n        // get text from textarea\r\n        let user_input = this.textarea.value;\r\n        // clear textarea\r\n        this.textarea.value = \"\";\r\n        // initiate response from assistant\r\n        this.initialize_response(user_input);\r\n      }\r\n      this.textarea.style.height = 'auto';\r\n      this.textarea.style.height = (this.textarea.scrollHeight) + 'px';\r\n    });\r\n    // button container\r\n    let button_container = chat_input.createDiv(\"sc-button-container\");\r\n    // create hidden abort button\r\n    let abort_button = button_container.createEl(\"span\", { attr: {id: \"sc-abort-button\", style: \"display: none;\"} });\r\n    Obsidian.setIcon(abort_button, \"square\");\r\n    // add event listener to button\r\n    abort_button.addEventListener(\"click\", () => {\r\n      // abort current response\r\n      this.end_stream();\r\n    });\r\n    // create button\r\n    let button = button_container.createEl(\"button\", { attr: {id: \"sc-send-button\"}, cls: \"send-button\" });\r\n    button.innerHTML = \"Send\";\r\n    // add event listener to button\r\n    button.addEventListener(\"click\", () => {\r\n      if(this.prevent_input){\r\n        console.log(\"wait until current response is finished\");\r\n        new Obsidian.Notice(\"Wait until current response is finished\");\r\n        return;\r\n      }\r\n      // get text from textarea\r\n      let user_input = this.textarea.value;\r\n      // clear textarea\r\n      this.textarea.value = \"\";\r\n      // initiate response from assistant\r\n      this.initialize_response(user_input);\r\n    });\r\n  }\r\n  async initialize_response(user_input) {\r\n    this.set_streaming_ux();\r\n    // render message\r\n    await this.render_message(user_input, \"user\");\r\n    this.chat.new_message_in_thread({\r\n      role: \"user\",\r\n      content: user_input\r\n    });\r\n    await this.render_dotdotdot();\r\n\r\n    // if contains internal link represented by [[link]]\r\n    if(this.chat.contains_internal_link(user_input)) {\r\n      this.chat.get_response_with_note_context(user_input, this);\r\n      return;\r\n    }\r\n    // // for testing purposes\r\n    // if(this.chat.contains_folder_reference(user_input)) {\r\n    //   const folders = this.chat.get_folder_references(user_input);\r\n    //   console.log(folders);\r\n    //   return;\r\n    // }\r\n    // if contains self referential keywords or folder reference\r\n    if(this.contains_self_referential_keywords(user_input) || this.chat.contains_folder_reference(user_input)) {\r\n      // get hyde\r\n      const context = await this.get_context_hyde(user_input);\r\n      // get user input with added context\r\n      // const context_input = this.build_context_input(context);\r\n      // console.log(context_input);\r\n      const chatml = [\r\n        {\r\n          role: \"system\",\r\n          // content: context_input\r\n          content: context\r\n        },\r\n        {\r\n          role: \"user\",\r\n          content: user_input\r\n        }\r\n      ];\r\n      this.request_chatgpt_completion({messages: chatml, temperature: 0});\r\n      return;\r\n    }\r\n    // completion without any specific context\r\n    this.request_chatgpt_completion();\r\n  }\r\n\r\n  async render_dotdotdot() {\r\n    if (this.dotdotdot_interval)\r\n      clearInterval(this.dotdotdot_interval);\r\n    await this.render_message(\"...\", \"assistant\");\r\n    // if is '...', then initiate interval to change to '.' and then to '..' and then to '...'\r\n    let dots = 0;\r\n    this.active_elm.innerHTML = '...';\r\n    this.dotdotdot_interval = setInterval(() => {\r\n      dots++;\r\n      if (dots > 3)\r\n        dots = 1;\r\n      this.active_elm.innerHTML = '.'.repeat(dots);\r\n    }, 500);\r\n    // wait 2 seconds for testing\r\n    // await new Promise(r => setTimeout(r, 2000));\r\n  }\r\n\r\n  set_streaming_ux() {\r\n    this.prevent_input = true;\r\n    // hide send button\r\n    if(document.getElementById(\"sc-send-button\"))\r\n      document.getElementById(\"sc-send-button\").style.display = \"none\";\r\n    // show abort button\r\n    if(document.getElementById(\"sc-abort-button\"))\r\n      document.getElementById(\"sc-abort-button\").style.display = \"block\";\r\n  }\r\n  unset_streaming_ux() {\r\n    this.prevent_input = false;\r\n    // show send button, remove display none\r\n    if(document.getElementById(\"sc-send-button\"))\r\n      document.getElementById(\"sc-send-button\").style.display = \"\";\r\n    // hide abort button\r\n    if(document.getElementById(\"sc-abort-button\"))\r\n      document.getElementById(\"sc-abort-button\").style.display = \"none\";\r\n  }\r\n\r\n\r\n  // check if includes keywords referring to one's own notes\r\n  contains_self_referential_keywords(user_input) {\r\n    const matches = user_input.match(this.plugin.self_ref_kw_regex);\r\n    if(matches) return true;\r\n    return false;\r\n  }\r\n\r\n  // render message\r\n  async render_message(message, from=\"assistant\", append_last=false) {\r\n    // if dotdotdot interval is set, then clear it\r\n    if(this.dotdotdot_interval) {\r\n      clearInterval(this.dotdotdot_interval);\r\n      this.dotdotdot_interval = null;\r\n      // clear last message\r\n      this.active_elm.innerHTML = '';\r\n    }\r\n    if(append_last) {\r\n      this.current_message_raw += message;\r\n      if(message.indexOf('\\n') === -1) {\r\n        this.active_elm.innerHTML += message;\r\n      }else{\r\n        this.active_elm.innerHTML = '';\r\n        // append to last message\r\n        await Obsidian.MarkdownRenderer.renderMarkdown(this.current_message_raw, this.active_elm, '?no-dataview', new Obsidian.Component());\r\n      }\r\n    }else{\r\n      this.current_message_raw = '';\r\n      if((this.chat.thread.length === 0) || (this.last_from !== from)) {\r\n        // create message\r\n        this.new_messsage_bubble(from);\r\n      }\r\n      // set message text\r\n      this.active_elm.innerHTML = '';\r\n      await Obsidian.MarkdownRenderer.renderMarkdown(message, this.active_elm, '?no-dataview', new Obsidian.Component());\r\n      // get links\r\n      this.handle_links_in_message();\r\n      // render button(s)\r\n      this.render_message_action_buttons(message);\r\n    }\r\n    // scroll to bottom\r\n    this.message_container.scrollTop = this.message_container.scrollHeight;\r\n  }\r\n  render_message_action_buttons(message) {\r\n    if (this.chat.context && this.chat.hyd) {\r\n      // render button to copy hyd in smart-connections code block\r\n      const context_view = this.active_elm.createEl(\"span\", {\r\n        cls: \"sc-msg-button\",\r\n        attr: {\r\n          title: \"Copy context to clipboard\" /* tooltip */\r\n        }\r\n      });\r\n      const this_hyd = this.chat.hyd;\r\n      Obsidian.setIcon(context_view, \"eye\");\r\n      context_view.addEventListener(\"click\", () => {\r\n        // copy to clipboard\r\n        navigator.clipboard.writeText(\"```smart-connections\\n\" + this_hyd + \"\\n```\\n\");\r\n        new Obsidian.Notice(\"[Smart Connections] Context code block copied to clipboard\");\r\n      });\r\n    }\r\n    if(this.chat.context) {\r\n      // render copy context button\r\n      const copy_prompt_button = this.active_elm.createEl(\"span\", {\r\n        cls: \"sc-msg-button\",\r\n        attr: {\r\n          title: \"Copy prompt to clipboard\" /* tooltip */\r\n        }\r\n      });\r\n      const this_context = this.chat.context.replace(/\\`\\`\\`/g, \"\\t```\").trimLeft();\r\n      Obsidian.setIcon(copy_prompt_button, \"files\");\r\n      copy_prompt_button.addEventListener(\"click\", () => {\r\n        // copy to clipboard\r\n        navigator.clipboard.writeText(\"```prompt-context\\n\" + this_context + \"\\n```\\n\");\r\n        new Obsidian.Notice(\"[Smart Connections] Context copied to clipboard\");\r\n      });\r\n    }\r\n    // render copy button\r\n    const copy_button = this.active_elm.createEl(\"span\", {\r\n      cls: \"sc-msg-button\",\r\n      attr: {\r\n        title: \"Copy message to clipboard\" /* tooltip */\r\n      }\r\n    });\r\n    Obsidian.setIcon(copy_button, \"copy\");\r\n    copy_button.addEventListener(\"click\", () => {\r\n      // copy message to clipboard\r\n      navigator.clipboard.writeText(message.trimLeft());\r\n      new Obsidian.Notice(\"[Smart Connections] Message copied to clipboard\");\r\n    });\r\n  }\r\n\r\n  handle_links_in_message() {\r\n    const links = this.active_elm.querySelectorAll(\"a\");\r\n    // if this active element contains a link\r\n    if (links.length > 0) {\r\n      for (let i = 0; i < links.length; i++) {\r\n        const link = links[i];\r\n        const link_text = link.getAttribute(\"data-href\");\r\n        // trigger hover event on link\r\n        link.addEventListener(\"mouseover\", (event) => {\r\n          this.app.workspace.trigger(\"hover-link\", {\r\n            event,\r\n            source: SMART_CONNECTIONS_CHAT_VIEW_TYPE,\r\n            hoverParent: link.parentElement,\r\n            targetEl: link,\r\n            // extract link text from a.data-href\r\n            linktext: link_text\r\n          });\r\n        });\r\n        // trigger open link event on link\r\n        link.addEventListener(\"click\", (event) => {\r\n          const link_tfile = this.app.metadataCache.getFirstLinkpathDest(link_text, \"/\");\r\n          // properly handle if the meta/ctrl key is pressed\r\n          const mod = Obsidian.Keymap.isModEvent(event);\r\n          // get most recent leaf\r\n          let leaf = this.app.workspace.getLeaf(mod);\r\n          leaf.openFile(link_tfile);\r\n        });\r\n      }\r\n    }\r\n  }\r\n\r\n  new_messsage_bubble(from) {\r\n    let message_el = this.message_container.createDiv(`sc-message ${from}`);\r\n    // create message content\r\n    this.active_elm = message_el.createDiv(\"sc-message-content\");\r\n    // set last from\r\n    this.last_from = from;\r\n  }\r\n\r\n  async request_chatgpt_completion(opts={}) {\r\n    const chat_ml = opts.messages || opts.chat_ml || this.chat.prepare_chat_ml();\r\n    console.log(\"chat_ml\", chat_ml);\r\n    const max_total_tokens = Math.round(get_max_chars(this.plugin.settings.smart_chat_model) / 4);\r\n    console.log(\"max_total_tokens\", max_total_tokens);\r\n    const curr_token_est = Math.round(JSON.stringify(chat_ml).length / 3);\r\n    console.log(\"curr_token_est\", curr_token_est);\r\n    let max_available_tokens = max_total_tokens - curr_token_est;\r\n    // if max_available_tokens is less than 0, set to 200\r\n    if(max_available_tokens < 0) max_available_tokens = 200;\r\n    else if(max_available_tokens > 4096) max_available_tokens = 4096;\r\n    console.log(\"max_available_tokens\", max_available_tokens);\r\n    opts = {\r\n      model: this.plugin.settings.smart_chat_model,\r\n      messages: chat_ml,\r\n      // max_tokens: 250,\r\n      max_tokens: max_available_tokens,\r\n      temperature: 0.3,\r\n      top_p: 1,\r\n      presence_penalty: 0,\r\n      frequency_penalty: 0,\r\n      stream: true,\r\n      stop: null,\r\n      n: 1,\r\n      // logit_bias: logit_bias,\r\n      ...opts\r\n    }\r\n    // console.log(opts.messages);\r\n    if(opts.stream) {\r\n      const full_str = await new Promise((resolve, reject) => {\r\n        try {\r\n          // console.log(\"stream\", opts);\r\n          const url = \"https://api.openai.com/v1/chat/completions\";\r\n          this.active_stream = new ScStreamer(url, {\r\n            headers: {\r\n              \"Content-Type\": \"application/json\",\r\n              Authorization: `Bearer ${this.plugin.settings.api_key}`\r\n            },\r\n            method: \"POST\",\r\n            payload: JSON.stringify(opts)\r\n          });\r\n          let txt = \"\";\r\n          this.active_stream.addEventListener(\"message\", (e) => {\r\n            if (e.data != \"[DONE]\") {\r\n              const payload = JSON.parse(e.data);\r\n              const text = payload.choices[0].delta.content;\r\n              if (!text) {\r\n                return;\r\n              }\r\n              txt += text;\r\n              this.render_message(text, \"assistant\", true);\r\n            } else {\r\n              this.end_stream();\r\n              resolve(txt);\r\n            }\r\n          });\r\n          this.active_stream.addEventListener(\"readystatechange\", (e) => {\r\n            if (e.readyState >= 2) {\r\n              console.log(\"ReadyState: \" + e.readyState);\r\n            }\r\n          });\r\n          this.active_stream.addEventListener(\"error\", (e) => {\r\n            console.error(e);\r\n            new Obsidian.Notice(\"Smart Connections Error Streaming Response. See console for details.\");\r\n            this.render_message(\"*API Error. See console logs for details.*\", \"assistant\");\r\n            this.end_stream();\r\n            reject(e);\r\n          });\r\n          this.active_stream.stream();\r\n        } catch (err) {\r\n          console.error(err);\r\n          new Obsidian.Notice(\"Smart Connections Error Streaming Response. See console for details.\");\r\n          this.end_stream();\r\n          reject(err);\r\n        }\r\n      });\r\n      // console.log(full_str);\r\n      await this.render_message(full_str, \"assistant\");\r\n      this.chat.new_message_in_thread({\r\n        role: \"assistant\",\r\n        content: full_str\r\n      });\r\n      return;\r\n    }else{\r\n      try{\r\n        const response = await (0, Obsidian.requestUrl)({\r\n          url: `https://api.openai.com/v1/chat/completions`,\r\n          method: \"POST\",\r\n          headers: {\r\n            Authorization: `Bearer ${this.plugin.settings.api_key}`,\r\n            \"Content-Type\": \"application/json\"\r\n          },\r\n          contentType: \"application/json\",\r\n          body: JSON.stringify(opts),\r\n          throw: false\r\n        });\r\n        // console.log(response);\r\n        return JSON.parse(response.text).choices[0].message.content;\r\n      }catch(err){\r\n        new Obsidian.Notice(`Smart Connections API Error :: ${err}`);\r\n      }\r\n    }\r\n  }\r\n\r\n  end_stream() {\r\n    if(this.active_stream){\r\n      this.active_stream.close();\r\n      this.active_stream = null;\r\n    }\r\n    this.unset_streaming_ux();\r\n    if(this.dotdotdot_interval){\r\n      clearInterval(this.dotdotdot_interval);\r\n      this.dotdotdot_interval = null;\r\n      // remove parent of active_elm\r\n      this.active_elm.parentElement.remove();\r\n      this.active_elm = null;\r\n    }\r\n  }\r\n\r\n  async get_context_hyde(user_input) {\r\n    this.chat.reset_context();\r\n    // count current chat ml messages to determine 'question' or 'chat log' wording\r\n    const hyd_input = `Anticipate what the user is seeking. Respond in the form of a hypothetical note written by the user. The note may contain statements as paragraphs, lists, or checklists in markdown format with no headings. Please respond with one hypothetical note and abstain from any other commentary. Use the format: PARENT FOLDER NAME > CHILD FOLDER NAME > FILE NAME > HEADING 1 > HEADING 2 > HEADING 3: HYPOTHETICAL NOTE CONTENTS.`;\r\n    // complete\r\n    const chatml = [\r\n      {\r\n        role: \"system\",\r\n        content: hyd_input\r\n      },\r\n      {\r\n        role: \"user\",\r\n        content: user_input\r\n      }\r\n    ];\r\n    const hyd = await this.request_chatgpt_completion({\r\n      messages: chatml,\r\n      stream: false,\r\n      temperature: 0,\r\n      max_tokens: 137,\r\n    });\r\n    this.chat.hyd = hyd;\r\n    // console.log(hyd);\r\n    let filter = {};\r\n    // if contains folder reference represented by /folder/\r\n    if(this.chat.contains_folder_reference(user_input)) {\r\n      // get folder references\r\n      const folder_refs = this.chat.get_folder_references(user_input);\r\n      // console.log(folder_refs);\r\n      // if folder references are valid (string or array of strings)\r\n      if(folder_refs){\r\n        filter = {\r\n          path_begins_with: folder_refs\r\n        };\r\n      }\r\n    }\r\n    // search for nearest based on hyd\r\n    let nearest = await this.plugin.api.search(hyd, filter);\r\n    console.log(\"nearest\", nearest.length);\r\n    nearest = this.get_nearest_until_next_dev_exceeds_std_dev(nearest);\r\n    console.log(\"nearest after std dev slice\", nearest.length);\r\n    nearest = this.sort_by_len_adjusted_similarity(nearest);\r\n\r\n    return await this.get_context_for_prompt(nearest);\r\n  }\r\n\r\n\r\n  sort_by_len_adjusted_similarity(nearest) {\r\n    // re-sort by quotient of similarity divided by len DESC\r\n    nearest = nearest.sort((a, b) => {\r\n      const a_score = a.similarity / a.len;\r\n      const b_score = b.similarity / b.len;\r\n      // if a is greater than b, return -1\r\n      if (a_score > b_score)\r\n        return -1;\r\n      // if a is less than b, return 1\r\n      if (a_score < b_score)\r\n        return 1;\r\n      // if a is equal to b, return 0\r\n      return 0;\r\n    });\r\n    return nearest;\r\n  }\r\n\r\n  get_nearest_until_next_dev_exceeds_std_dev(nearest) {\r\n    // get std dev of similarity\r\n    const sim = nearest.map((n) => n.similarity);\r\n    const mean = sim.reduce((a, b) => a + b) / sim.length;\r\n    let std_dev = Math.sqrt(sim.map((x) => Math.pow(x - mean, 2)).reduce((a, b) => a + b) / sim.length);\r\n    // slice where next item deviation is greater than std_dev\r\n    let slice_i = 0;\r\n    while (slice_i < nearest.length) {\r\n      const next = nearest[slice_i + 1];\r\n      if (next) {\r\n        const next_dev = Math.abs(next.similarity - nearest[slice_i].similarity);\r\n        if (next_dev > std_dev) {\r\n          if(slice_i < 3) std_dev = std_dev * 1.5;\r\n          else break;\r\n        }\r\n      }\r\n      slice_i++;\r\n    }\r\n    // select top results\r\n    nearest = nearest.slice(0, slice_i+1);\r\n    return nearest;\r\n  }\r\n  // this.test_get_nearest_until_next_dev_exceeds_std_dev();\r\n  // // test get_nearest_until_next_dev_exceeds_std_dev\r\n  // test_get_nearest_until_next_dev_exceeds_std_dev() {\r\n  //   const nearest = [{similarity: 0.99}, {similarity: 0.98}, {similarity: 0.97}, {similarity: 0.96}, {similarity: 0.95}, {similarity: 0.94}, {similarity: 0.93}, {similarity: 0.92}, {similarity: 0.91}, {similarity: 0.9}, {similarity: 0.79}, {similarity: 0.78}, {similarity: 0.77}, {similarity: 0.76}, {similarity: 0.75}, {similarity: 0.74}, {similarity: 0.73}, {similarity: 0.72}];\r\n  //   const result = this.get_nearest_until_next_dev_exceeds_std_dev(nearest);\r\n  //   if(result.length !== 10){\r\n  //     console.error(\"get_nearest_until_next_dev_exceeds_std_dev failed\", result);\r\n  //   }\r\n  // }\r\n\r\n  async get_context_for_prompt(nearest) {\r\n    let context = [];\r\n    const MAX_SOURCES = (this.plugin.settings.smart_chat_model === 'gpt-4-1106-preview') ? 42 : 20;\r\n    const MAX_CHARS = get_max_chars(this.plugin.settings.smart_chat_model) / 2;\r\n    let char_accum = 0;\r\n    for (let i = 0; i < nearest.length; i++) {\r\n      if (context.length >= MAX_SOURCES)\r\n        break;\r\n      if (char_accum >= MAX_CHARS)\r\n        break;\r\n      if (typeof nearest[i].link !== 'string')\r\n        continue;\r\n      // generate breadcrumbs\r\n      const breadcrumbs = nearest[i].link.replace(/#/g, \" > \").replace(\".md\", \"\").replace(/\\//g, \" > \");\r\n      let new_context = `${breadcrumbs}:\\n`;\r\n      // get max available chars to add to context\r\n      const max_available_chars = MAX_CHARS - char_accum - new_context.length;\r\n      if (nearest[i].link.indexOf(\"#\") !== -1) { // is block\r\n        new_context += await this.plugin.block_retriever(nearest[i].link, { max_chars: max_available_chars });\r\n      } else { // is file\r\n        new_context += await this.plugin.file_retriever(nearest[i].link, { max_chars: max_available_chars });\r\n      }\r\n      // add to char_accum\r\n      char_accum += new_context.length;\r\n      // add to context\r\n      context.push({\r\n        link: nearest[i].link,\r\n        text: new_context\r\n      });\r\n    }\r\n    // context sources\r\n    console.log(\"context sources: \" + context.length);\r\n    // char_accum divided by 4 and rounded to nearest integer for estimated tokens\r\n    console.log(\"total context tokens: ~\" + Math.round(char_accum / 3.5));\r\n    // build context input\r\n    this.chat.context = `Anticipate the type of answer desired by the user. Imagine the following ${context.length} notes were written by the user and contain all the necessary information to answer the user's question. Begin responses with \"${SMART_TRANSLATION[this.plugin.settings.language].prompt}...\"`;\r\n    for(let i = 0; i < context.length; i++) {\r\n      this.chat.context += `\\n---BEGIN #${i+1}---\\n${context[i].text}\\n---END #${i+1}---`;\r\n    }\r\n    return this.chat.context;\r\n  }\r\n\r\n\r\n}\r\n\r\nfunction get_max_chars(model=\"gpt-3.5-turbo\") {\r\n  const MAX_CHAR_MAP = {\r\n    \"gpt-3.5-turbo-16k\": 48000,\r\n    \"gpt-4\": 24000,\r\n    \"gpt-3.5-turbo\": 12000,\r\n    \"gpt-4-1106-preview\": 200000,\r\n  };\r\n  return MAX_CHAR_MAP[model];\r\n}\r\n/**\r\n * SmartConnectionsChatModel\r\n * ---\r\n * - 'thread' format: Array[Array[Object{role, content, hyde}]]\r\n *  - [Turn[variation{}], Turn[variation{}, variation{}], ...]\r\n * - Saves in 'thread' format to JSON file in .smart-connections folder using chat_id as filename\r\n * - Loads chat in 'thread' format Array[Array[Object{role, content, hyde}]] from JSON file in .smart-connections folder\r\n * - prepares chat_ml returns in 'ChatML' format\r\n *  - strips all but role and content properties from Object in ChatML format\r\n * - ChatML Array[Object{role, content}]\r\n *  - [Current_Variation_For_Turn_1{}, Current_Variation_For_Turn_2{}, ...]\r\n */\r\nclass SmartConnectionsChatModel {\r\n  constructor(plugin) {\r\n    this.app = plugin.app;\r\n    this.plugin = plugin;\r\n    this.chat_id = null;\r\n    this.chat_ml = [];\r\n    this.context = null;\r\n    this.hyd = null;\r\n    this.thread = [];\r\n  }\r\n  async save_chat() {\r\n    // return if thread is empty\r\n    if (this.thread.length === 0) return;\r\n    // save chat to file in .smart-connections folder\r\n    // create .smart-connections/chats/ folder if it doesn't exist\r\n    if (!(await this.app.vault.adapter.exists(\".smart-connections/chats\"))) {\r\n      await this.app.vault.adapter.mkdir(\".smart-connections/chats\");\r\n    }\r\n    // if chat_id is not set, set it to UNTITLED-${unix timestamp}\r\n    if (!this.chat_id) {\r\n      this.chat_id = this.name() + \"\u2014\" + this.get_file_date_string();\r\n    }\r\n    // validate chat_id is set to valid filename characters (letters, numbers, underscores, dashes, em dash, and spaces)\r\n    if (!this.chat_id.match(/^[a-zA-Z0-9_\u2014\\- ]+$/)) {\r\n      console.log(\"Invalid chat_id: \" + this.chat_id);\r\n      new Obsidian.Notice(\"[Smart Connections] Failed to save chat. Invalid chat_id: '\" + this.chat_id + \"'\");\r\n    }\r\n    // filename is chat_id\r\n    const chat_file = this.chat_id + \".json\";\r\n    this.app.vault.adapter.write(\r\n      \".smart-connections/chats/\" + chat_file,\r\n      JSON.stringify(this.thread, null, 2)\r\n    );\r\n  }\r\n  async load_chat(chat_id) {\r\n    this.chat_id = chat_id;\r\n    // load chat from file in .smart-connections folder\r\n    // filename is chat_id\r\n    const chat_file = this.chat_id + \".json\";\r\n    // read file\r\n    let chat_json = await this.app.vault.adapter.read(\r\n      \".smart-connections/chats/\" + chat_file\r\n    );\r\n    // parse json\r\n    this.thread = JSON.parse(chat_json);\r\n    // load chat_ml\r\n    this.chat_ml = this.prepare_chat_ml();\r\n    // render messages in chat view\r\n    // for each turn in chat_ml\r\n    // console.log(this.thread);\r\n    // console.log(this.chat_ml);\r\n  }\r\n  // prepare chat_ml from chat\r\n  // gets the last message of each turn unless turn_variation_offsets=[[turn_index,variation_index]] is specified in offset\r\n  prepare_chat_ml(turn_variation_offsets=[]) {\r\n    // if no turn_variation_offsets, get the last message of each turn\r\n    if(turn_variation_offsets.length === 0){\r\n      this.chat_ml = this.thread.map(turn => {\r\n        return turn[turn.length - 1];\r\n      });\r\n    }else{\r\n      // create an array from turn_variation_offsets that indexes variation_index at turn_index\r\n      // ex. [[3,5]] => [undefined, undefined, undefined, 5]\r\n      let turn_variation_index = [];\r\n      for(let i = 0; i < turn_variation_offsets.length; i++){\r\n        turn_variation_index[turn_variation_offsets[i][0]] = turn_variation_offsets[i][1];\r\n      }\r\n      // loop through chat\r\n      this.chat_ml = this.thread.map((turn, turn_index) => {\r\n        // if there is an index for this turn, return the variation at that index\r\n        if(turn_variation_index[turn_index] !== undefined){\r\n          return turn[turn_variation_index[turn_index]];\r\n        }\r\n        // otherwise return the last message of the turn\r\n        return turn[turn.length - 1];\r\n      });\r\n    }\r\n    // strip all but role and content properties from each message\r\n    this.chat_ml = this.chat_ml.map(message => {\r\n      return {\r\n        role: message.role,\r\n        content: message.content\r\n      };\r\n    });\r\n    return this.chat_ml;\r\n  }\r\n  last() {\r\n    // get last message from chat\r\n    return this.thread[this.thread.length - 1][this.thread[this.thread.length - 1].length - 1];\r\n  }\r\n  last_from() {\r\n    return this.last().role;\r\n  }\r\n  // returns user_input or completion\r\n  last_message() {\r\n    return this.last().content;\r\n  }\r\n  // message={}\r\n  // add new message to thread\r\n  new_message_in_thread(message, turn=-1) {\r\n    // if turn is -1, add to new turn\r\n    if(this.context){\r\n      message.context = this.context;\r\n      this.context = null;\r\n    }\r\n    if(this.hyd){\r\n      message.hyd = this.hyd;\r\n      this.hyd = null;\r\n    }\r\n    if (turn === -1) {\r\n      this.thread.push([message]);\r\n    }else{\r\n      // otherwise add to specified turn\r\n      this.thread[turn].push(message);\r\n    }\r\n  }\r\n  reset_context(){\r\n    this.context = null;\r\n    this.hyd = null;\r\n  }\r\n  async rename_chat(new_name){\r\n    // check if current chat_id file exists\r\n    if (this.chat_id && await this.app.vault.adapter.exists(\".smart-connections/chats/\" + this.chat_id + \".json\")) {\r\n      new_name = this.chat_id.replace(this.name(), new_name);\r\n      // rename file if it exists\r\n      await this.app.vault.adapter.rename(\r\n        \".smart-connections/chats/\" + this.chat_id + \".json\",\r\n        \".smart-connections/chats/\" + new_name + \".json\"\r\n      );\r\n      // set chat_id to new_name\r\n      this.chat_id = new_name;\r\n    }else{\r\n      this.chat_id = new_name + \"\u2014\" + this.get_file_date_string();\r\n      // save chat\r\n      await this.save_chat();\r\n    }\r\n\r\n  }\r\n\r\n  name() {\r\n    if(this.chat_id){\r\n      // remove date after last em dash\r\n      return this.chat_id.replace(/\u2014[^\u2014]*$/,\"\");\r\n    }\r\n    return \"UNTITLED\";\r\n  }\r\n\r\n  get_file_date_string() {\r\n    return new Date().toISOString().replace(/(T|:|\\..*)/g, \" \").trim();\r\n  }\r\n  // get response from with note context\r\n  async get_response_with_note_context(user_input, chat_view) {\r\n    let system_input = \"Imagine the following notes were written by the user and contain the necessary information to synthesize a useful answer the user's query:\\n\";\r\n    // extract internal links\r\n    const notes = this.extract_internal_links(user_input);\r\n    // get content of internal links as context\r\n    let max_chars = get_max_chars(this.plugin.settings.smart_chat_model);\r\n    for(let i = 0; i < notes.length; i++){\r\n      // max chars for this note is max_chars divided by number of notes left\r\n      const this_max_chars = (notes.length - i > 1) ? Math.floor(max_chars / (notes.length - i)) : max_chars;\r\n      // console.log(\"file context max chars: \" + this_max_chars);\r\n      const note_content = await this.get_note_contents(notes[i], {char_limit: this_max_chars});\r\n      console.log(note_content);\r\n      system_input += `---BEGIN NOTE: [[${notes[i].basename}]]---\\n`\r\n      system_input += note_content;\r\n      system_input += `---END NOTE---\\n`\r\n      max_chars -= note_content.length;\r\n      if(max_chars <= 0) break;\r\n    }\r\n    this.context = system_input;\r\n    const chatml = [\r\n      {\r\n        role: \"system\",\r\n        content: system_input\r\n      },\r\n      {\r\n        role: \"user\",\r\n        content: user_input\r\n      }\r\n    ];\r\n    chat_view.request_chatgpt_completion({messages: chatml, temperature: 0});\r\n  }\r\n  // check if contains internal link\r\n  contains_internal_link(user_input) {\r\n    if(user_input.indexOf(\"[[\") === -1) return false;\r\n    if(user_input.indexOf(\"]]\") === -1) return false;\r\n    return true;\r\n  }\r\n  // check if contains folder reference (ex. /folder/, or /folder/subfolder/)\r\n  contains_folder_reference(user_input) {\r\n    if(user_input.indexOf(\"/\") === -1) return false;\r\n    if(user_input.indexOf(\"/\") === user_input.lastIndexOf(\"/\")) return false;\r\n    return true;\r\n  }\r\n  // get folder references from user input\r\n  get_folder_references(user_input) {\r\n    // use this.folders to extract folder references by longest first (ex. /folder/subfolder/ before /folder/) to avoid matching /folder/subfolder/ as /folder/\r\n    const folders = this.plugin.folders.slice(); // copy folders array\r\n    const matches = folders.sort((a, b) => b.length - a.length).map(folder => {\r\n      // check if folder is in user_input\r\n      if(user_input.indexOf(folder) !== -1){\r\n        // remove folder from user_input to prevent matching /folder/subfolder/ as /folder/\r\n        user_input = user_input.replace(folder, \"\");\r\n        return folder;\r\n      }\r\n      return false;\r\n    }).filter(folder => folder);\r\n    console.log(matches);\r\n    // return array of matches\r\n    if(matches) return matches;\r\n    return false;\r\n  }\r\n\r\n\r\n  // extract internal links\r\n  extract_internal_links(user_input) {\r\n    const matches = user_input.match(/\\[\\[(.*?)\\]\\]/g);\r\n    console.log(matches);\r\n    // return array of TFile objects\r\n    if(matches) return matches.map(match => {\r\n      return this.app.metadataCache.getFirstLinkpathDest(match.replace(\"[[\", \"\").replace(\"]]\", \"\"), \"/\");\r\n    });\r\n    return [];\r\n  }\r\n  // get context from internal links\r\n  async get_note_contents(note, opts={}) {\r\n    opts = {\r\n      char_limit: 10000,\r\n      ...opts\r\n    }\r\n    // return if note is not a file\r\n    if(!(note instanceof Obsidian.TFile)) return \"\";\r\n    // get file content\r\n    let file_content = await this.app.vault.cachedRead(note);\r\n    //cut off front matter\r\n    if (this.plugin.settings.cut_off_frontmatter) {\r\n      file_content = file_content.replace(/\\s*---[\\s\\S]*?---/,\"\");\r\n    }\r\n    // check if contains dataview code block\r\n    if(file_content.indexOf(\"```dataview\") > -1){\r\n      // if contains dataview code block get all dataview code blocks\r\n      file_content = await this.render_dataview_queries(file_content, note.path, opts);\r\n    }\r\n    return file_content.substring(0, opts.char_limit);\r\n  }\r\n\r\n\r\n  async render_dataview_queries(file_content, note_path, opts={}) {\r\n    opts = {\r\n      char_limit: null,\r\n      ...opts\r\n    };\r\n    // use window to get dataview api\r\n    const dataview_api = window[\"DataviewAPI\"];\r\n    // skip if dataview api not found\r\n    if(!dataview_api) return file_content;\r\n    const dataview_code_blocks = file_content.match(/```dataview(.*?)```/gs);\r\n    // for each dataview code block\r\n    for (let i = 0; i < dataview_code_blocks.length; i++) {\r\n      // if opts char_limit is less than indexOf dataview code block, break\r\n      if(opts.char_limit && opts.char_limit < file_content.indexOf(dataview_code_blocks[i])) break;\r\n      // get dataview code block\r\n      const dataview_code_block = dataview_code_blocks[i];\r\n      // get content of dataview code block\r\n      const dataview_code_block_content = dataview_code_block.replace(\"```dataview\", \"\").replace(\"```\", \"\");\r\n      // get dataview query result\r\n      const dataview_query_result = await dataview_api.queryMarkdown(dataview_code_block_content, note_path, null);\r\n      // if query result is successful, replace dataview code block with query result\r\n      if (dataview_query_result.successful) {\r\n        file_content = file_content.replace(dataview_code_block, dataview_query_result.value);\r\n      }\r\n    }\r\n    return file_content;\r\n  }\r\n}\r\n\r\nclass SmartConnectionsChatHistoryModal extends Obsidian.FuzzySuggestModal {\r\n  constructor(app, view, files) {\r\n    super(app);\r\n    this.app = app;\r\n    this.view = view;\r\n    this.setPlaceholder(\"Type the name of a chat session...\");\r\n  }\r\n  getItems() {\r\n    if (!this.view.files) {\r\n      return [];\r\n    }\r\n    return this.view.files;\r\n  }\r\n  getItemText(item) {\r\n    // if not UNTITLED, remove date after last em dash\r\n    if(item.indexOf(\"UNTITLED\") === -1){\r\n      item.replace(/\u2014[^\u2014]*$/,\"\");\r\n    }\r\n    return item;\r\n  }\r\n  onChooseItem(session) {\r\n    this.view.open_chat(session);\r\n  }\r\n}\r\n\r\n// File Select Fuzzy Suggest Modal\r\nclass SmartConnectionsFileSelectModal extends Obsidian.FuzzySuggestModal {\r\n  constructor(app, view) {\r\n    super(app);\r\n    this.app = app;\r\n    this.view = view;\r\n    this.setPlaceholder(\"Type the name of a file...\");\r\n  }\r\n  getItems() {\r\n    // get all markdown files\r\n    return this.app.vault.getMarkdownFiles().sort((a, b) => a.basename.localeCompare(b.basename));\r\n  }\r\n  getItemText(item) {\r\n    return item.basename;\r\n  }\r\n  onChooseItem(file) {\r\n    this.view.insert_selection(file.basename + \"]] \");\r\n  }\r\n}\r\n// Folder Select Fuzzy Suggest Modal\r\nclass SmartConnectionsFolderSelectModal extends Obsidian.FuzzySuggestModal {\r\n  constructor(app, view) {\r\n    super(app);\r\n    this.app = app;\r\n    this.view = view;\r\n    this.setPlaceholder(\"Type the name of a folder...\");\r\n  }\r\n  getItems() {\r\n    return this.view.plugin.folders;\r\n  }\r\n  getItemText(item) {\r\n    return item;\r\n  }\r\n  onChooseItem(folder) {\r\n    this.view.insert_selection(folder + \"/ \");\r\n  }\r\n}\r\n\r\n\r\n// Handle API response streaming\r\nclass ScStreamer {\r\n  // constructor\r\n  constructor(url, options) {\r\n    // set default options\r\n    options = options || {};\r\n    this.url = url;\r\n    this.method = options.method || 'GET';\r\n    this.headers = options.headers || {};\r\n    this.payload = options.payload || null;\r\n    this.withCredentials = options.withCredentials || false;\r\n    this.listeners = {};\r\n    this.readyState = this.CONNECTING;\r\n    this.progress = 0;\r\n    this.chunk = '';\r\n    this.xhr = null;\r\n    this.FIELD_SEPARATOR = ':';\r\n    this.INITIALIZING = -1;\r\n    this.CONNECTING = 0;\r\n    this.OPEN = 1;\r\n    this.CLOSED = 2;\r\n  }\r\n  // addEventListener\r\n  addEventListener(type, listener) {\r\n    // check if the type is in the listeners\r\n    if (!this.listeners[type]) {\r\n      this.listeners[type] = [];\r\n    }\r\n    // check if the listener is already in the listeners\r\n    if(this.listeners[type].indexOf(listener) === -1) {\r\n      this.listeners[type].push(listener);\r\n    }\r\n  }\r\n  // removeEventListener\r\n  removeEventListener(type, listener) {\r\n    // check if listener type is undefined\r\n    if (!this.listeners[type]) {\r\n      return;\r\n    }\r\n    let filtered = [];\r\n    // loop through the listeners\r\n    for (let i = 0; i < this.listeners[type].length; i++) {\r\n      // check if the listener is the same\r\n      if (this.listeners[type][i] !== listener) {\r\n        filtered.push(this.listeners[type][i]);\r\n      }\r\n    }\r\n    // check if the listeners are empty\r\n    if (this.listeners[type].length === 0) {\r\n      delete this.listeners[type];\r\n    } else {\r\n      this.listeners[type] = filtered;\r\n    }\r\n  }\r\n  // dispatchEvent\r\n  dispatchEvent(event) {\r\n    // if no event return true\r\n    if (!event) {\r\n      return true;\r\n    }\r\n    // set event source to this\r\n    event.source = this;\r\n    // set onHandler to on + event type\r\n    let onHandler = 'on' + event.type;\r\n    // check if the onHandler has own property named same as onHandler\r\n    if (this.hasOwnProperty(onHandler)) {\r\n      // call the onHandler\r\n      this[onHandler].call(this, event);\r\n      // check if the event is default prevented\r\n      if (event.defaultPrevented) {\r\n        return false;\r\n      }\r\n    }\r\n    // check if the event type is in the listeners\r\n    if (this.listeners[event.type]) {\r\n      return this.listeners[event.type].every(function(callback) {\r\n        callback(event);\r\n        return !event.defaultPrevented;\r\n      });\r\n    }\r\n    return true;\r\n  }\r\n  // _setReadyState\r\n  _setReadyState(state) {\r\n    // set event type to readyStateChange\r\n    let event = new CustomEvent('readyStateChange');\r\n    // set event readyState to state\r\n    event.readyState = state;\r\n    // set readyState to state\r\n    this.readyState = state;\r\n    // dispatch event\r\n    this.dispatchEvent(event);\r\n  }\r\n  // _onStreamFailure\r\n  _onStreamFailure(e) {\r\n    // set event type to error\r\n    let event = new CustomEvent('error');\r\n    // set event data to e\r\n    event.data = e.currentTarget.response;\r\n    // dispatch event\r\n    this.dispatchEvent(event);\r\n    this.close();\r\n  }\r\n  // _onStreamAbort\r\n  _onStreamAbort(e) {\r\n    // set to abort\r\n    let event = new CustomEvent('abort');\r\n    // close\r\n    this.close();\r\n  }\r\n  // _onStreamProgress\r\n  _onStreamProgress(e) {\r\n    // if not xhr return\r\n    if (!this.xhr) {\r\n      return;\r\n    }\r\n    // if xhr status is not 200 return\r\n    if (this.xhr.status !== 200) {\r\n      // onStreamFailure\r\n      this._onStreamFailure(e);\r\n      return;\r\n    }\r\n    // if ready state is CONNECTING\r\n    if (this.readyState === this.CONNECTING) {\r\n      // dispatch event\r\n      this.dispatchEvent(new CustomEvent('open'));\r\n      // set ready state to OPEN\r\n      this._setReadyState(this.OPEN);\r\n    }\r\n    // parse the received data.\r\n    let data = this.xhr.responseText.substring(this.progress);\r\n    // update progress\r\n    this.progress += data.length;\r\n    // split the data by new line and parse each line\r\n    data.split(/(\\r\\n|\\r|\\n){2}/g).forEach(function(part){\r\n      if(part.trim().length === 0) {\r\n        this.dispatchEvent(this._parseEventChunk(this.chunk.trim()));\r\n        this.chunk = '';\r\n      } else {\r\n        this.chunk += part;\r\n      }\r\n    }.bind(this));\r\n  }\r\n  // _onStreamLoaded\r\n  _onStreamLoaded(e) {\r\n    this._onStreamProgress(e);\r\n    // parse the last chunk\r\n    this.dispatchEvent(this._parseEventChunk(this.chunk));\r\n    this.chunk = '';\r\n  }\r\n  // _parseEventChunk\r\n  _parseEventChunk(chunk) {\r\n    // if no chunk or chunk is empty return\r\n    if (!chunk || chunk.length === 0) {\r\n      return null;\r\n    }\r\n    // init e\r\n    let e = {id: null, retry: null, data: '', event: 'message'};\r\n    // split the chunk by new line\r\n    chunk.split(/(\\r\\n|\\r|\\n)/).forEach(function(line) {\r\n      line = line.trimRight();\r\n      let index = line.indexOf(this.FIELD_SEPARATOR);\r\n      if(index <= 0) {\r\n        return;\r\n      }\r\n      // field\r\n      let field = line.substring(0, index);\r\n      if(!(field in e)) {\r\n        return;\r\n      }\r\n      // value\r\n      let value = line.substring(index + 1).trimLeft();\r\n      if(field === 'data') {\r\n        e[field] += value;\r\n      } else {\r\n        e[field] = value;\r\n      }\r\n    }.bind(this));\r\n    // return event\r\n    let event = new CustomEvent(e.event);\r\n    event.data = e.data;\r\n    event.id = e.id;\r\n    return event;\r\n  }\r\n  // _checkStreamClosed\r\n  _checkStreamClosed() {\r\n    if(!this.xhr) {\r\n      return;\r\n    }\r\n    if(this.xhr.readyState === XMLHttpRequest.DONE) {\r\n      this._setReadyState(this.CLOSED);\r\n    }\r\n  }\r\n  // stream\r\n  stream() {\r\n    // set ready state to connecting\r\n    this._setReadyState(this.CONNECTING);\r\n    // set xhr to new XMLHttpRequest\r\n    this.xhr = new XMLHttpRequest();\r\n    // set xhr progress to _onStreamProgress\r\n    this.xhr.addEventListener('progress', this._onStreamProgress.bind(this));\r\n    // set xhr load to _onStreamLoaded\r\n    this.xhr.addEventListener('load', this._onStreamLoaded.bind(this));\r\n    // set xhr ready state change to _checkStreamClosed\r\n    this.xhr.addEventListener('readystatechange', this._checkStreamClosed.bind(this));\r\n    // set xhr error to _onStreamFailure\r\n    this.xhr.addEventListener('error', this._onStreamFailure.bind(this));\r\n    // set xhr abort to _onStreamAbort\r\n    this.xhr.addEventListener('abort', this._onStreamAbort.bind(this));\r\n    // open xhr\r\n    this.xhr.open(this.method, this.url);\r\n    // headers to xhr\r\n    for (let header in this.headers) {\r\n      this.xhr.setRequestHeader(header, this.headers[header]);\r\n    }\r\n    // credentials to xhr\r\n    this.xhr.withCredentials = this.withCredentials;\r\n    // send xhr\r\n    this.xhr.send(this.payload);\r\n  }\r\n  // close\r\n  close() {\r\n    if(this.readyState === this.CLOSED) {\r\n      return;\r\n    }\r\n    this.xhr.abort();\r\n    this.xhr = null;\r\n    this._setReadyState(this.CLOSED);\r\n  }\r\n}\r\n\r\nmodule.exports = SmartConnectionsPlugin;"],
  "mappings": "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",
  "names": ["line_limit", "item", "link", "file_link", "file_link_list"]
}
 diff --git a/package.json b/package.json index 2a94e90d..6a3e9c47 100644 --- a/package.json +++ b/package.json @@ -27,8 +27,5 @@ "homepage": "https://github.com/brianpetro/obsidian-smart-connections", "devDependencies": { "esbuild": "^0.17.19" - }, - "dependencies": { - "vec-lite": "file:../vec_lite" } } diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml new file mode 100644 index 00000000..1ab1953a --- /dev/null +++ b/pnpm-lock.yaml @@ -0,0 +1,240 @@ +lockfileVersion: '6.0' + +settings: + autoInstallPeers: true + excludeLinksFromLockfile: false + +devDependencies: + esbuild: + specifier: ^0.17.19 + version: 0.17.19 + +packages: + + /@esbuild/android-arm64@0.17.19: + resolution: {integrity: sha512-KBMWvEZooR7+kzY0BtbTQn0OAYY7CsiydT63pVEaPtVYF0hXbUaOyZog37DKxK7NF3XacBJOpYT4adIJh+avxA==} + engines: {node: '>=12'} + cpu: [arm64] + os: [android] + requiresBuild: true + dev: true + optional: true + + /@esbuild/android-arm@0.17.19: + resolution: {integrity: sha512-rIKddzqhmav7MSmoFCmDIb6e2W57geRsM94gV2l38fzhXMwq7hZoClug9USI2pFRGL06f4IOPHHpFNOkWieR8A==} + engines: {node: '>=12'} + cpu: [arm] + os: [android] + requiresBuild: true + dev: true + optional: true + + /@esbuild/android-x64@0.17.19: + resolution: {integrity: sha512-uUTTc4xGNDT7YSArp/zbtmbhO0uEEK9/ETW29Wk1thYUJBz3IVnvgEiEwEa9IeLyvnpKrWK64Utw2bgUmDveww==} + engines: {node: '>=12'} + cpu: [x64] + os: [android] + requiresBuild: true + dev: true + optional: true + + /@esbuild/darwin-arm64@0.17.19: + resolution: {integrity: sha512-80wEoCfF/hFKM6WE1FyBHc9SfUblloAWx6FJkFWTWiCoht9Mc0ARGEM47e67W9rI09YoUxJL68WHfDRYEAvOhg==} + engines: {node: '>=12'} + cpu: [arm64] + os: [darwin] + requiresBuild: true + dev: true + optional: true + + /@esbuild/darwin-x64@0.17.19: + resolution: {integrity: sha512-IJM4JJsLhRYr9xdtLytPLSH9k/oxR3boaUIYiHkAawtwNOXKE8KoU8tMvryogdcT8AU+Bflmh81Xn6Q0vTZbQw==} + engines: {node: '>=12'} + cpu: [x64] + os: [darwin] + requiresBuild: true + dev: true + optional: true + + /@esbuild/freebsd-arm64@0.17.19: + resolution: {integrity: sha512-pBwbc7DufluUeGdjSU5Si+P3SoMF5DQ/F/UmTSb8HXO80ZEAJmrykPyzo1IfNbAoaqw48YRpv8shwd1NoI0jcQ==} + engines: {node: '>=12'} + cpu: [arm64] + os: [freebsd] + requiresBuild: true + dev: true + optional: true + + /@esbuild/freebsd-x64@0.17.19: + resolution: {integrity: sha512-4lu+n8Wk0XlajEhbEffdy2xy53dpR06SlzvhGByyg36qJw6Kpfk7cp45DR/62aPH9mtJRmIyrXAS5UWBrJT6TQ==} + engines: {node: '>=12'} + cpu: [x64] + os: [freebsd] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-arm64@0.17.19: + resolution: {integrity: sha512-ct1Tg3WGwd3P+oZYqic+YZF4snNl2bsnMKRkb3ozHmnM0dGWuxcPTTntAF6bOP0Sp4x0PjSF+4uHQ1xvxfRKqg==} + engines: {node: '>=12'} + cpu: [arm64] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-arm@0.17.19: + resolution: {integrity: sha512-cdmT3KxjlOQ/gZ2cjfrQOtmhG4HJs6hhvm3mWSRDPtZ/lP5oe8FWceS10JaSJC13GBd4eH/haHnqf7hhGNLerA==} + engines: {node: '>=12'} + cpu: [arm] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-ia32@0.17.19: + resolution: {integrity: sha512-w4IRhSy1VbsNxHRQpeGCHEmibqdTUx61Vc38APcsRbuVgK0OPEnQ0YD39Brymn96mOx48Y2laBQGqgZ0j9w6SQ==} + engines: {node: '>=12'} + cpu: [ia32] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-loong64@0.17.19: + resolution: {integrity: sha512-2iAngUbBPMq439a+z//gE+9WBldoMp1s5GWsUSgqHLzLJ9WoZLZhpwWuym0u0u/4XmZ3gpHmzV84PonE+9IIdQ==} + engines: {node: '>=12'} + cpu: [loong64] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-mips64el@0.17.19: + resolution: {integrity: sha512-LKJltc4LVdMKHsrFe4MGNPp0hqDFA1Wpt3jE1gEyM3nKUvOiO//9PheZZHfYRfYl6AwdTH4aTcXSqBerX0ml4A==} + engines: {node: '>=12'} + cpu: [mips64el] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-ppc64@0.17.19: + resolution: {integrity: sha512-/c/DGybs95WXNS8y3Ti/ytqETiW7EU44MEKuCAcpPto3YjQbyK3IQVKfF6nbghD7EcLUGl0NbiL5Rt5DMhn5tg==} + engines: {node: '>=12'} + cpu: [ppc64] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-riscv64@0.17.19: + resolution: {integrity: sha512-FC3nUAWhvFoutlhAkgHf8f5HwFWUL6bYdvLc/TTuxKlvLi3+pPzdZiFKSWz/PF30TB1K19SuCxDTI5KcqASJqA==} + engines: {node: '>=12'} + cpu: [riscv64] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-s390x@0.17.19: + resolution: {integrity: sha512-IbFsFbxMWLuKEbH+7sTkKzL6NJmG2vRyy6K7JJo55w+8xDk7RElYn6xvXtDW8HCfoKBFK69f3pgBJSUSQPr+4Q==} + engines: {node: '>=12'} + cpu: [s390x] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-x64@0.17.19: + resolution: {integrity: sha512-68ngA9lg2H6zkZcyp22tsVt38mlhWde8l3eJLWkyLrp4HwMUr3c1s/M2t7+kHIhvMjglIBrFpncX1SzMckomGw==} + engines: {node: '>=12'} + cpu: [x64] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/netbsd-x64@0.17.19: + resolution: {integrity: sha512-CwFq42rXCR8TYIjIfpXCbRX0rp1jo6cPIUPSaWwzbVI4aOfX96OXY8M6KNmtPcg7QjYeDmN+DD0Wp3LaBOLf4Q==} + engines: {node: '>=12'} + cpu: [x64] + os: [netbsd] + requiresBuild: true + dev: true + optional: true + + /@esbuild/openbsd-x64@0.17.19: + resolution: {integrity: sha512-cnq5brJYrSZ2CF6c35eCmviIN3k3RczmHz8eYaVlNasVqsNY+JKohZU5MKmaOI+KkllCdzOKKdPs762VCPC20g==} + engines: {node: '>=12'} + cpu: [x64] + os: [openbsd] + requiresBuild: true + dev: true + optional: true + + /@esbuild/sunos-x64@0.17.19: + resolution: {integrity: sha512-vCRT7yP3zX+bKWFeP/zdS6SqdWB8OIpaRq/mbXQxTGHnIxspRtigpkUcDMlSCOejlHowLqII7K2JKevwyRP2rg==} + engines: {node: '>=12'} + cpu: [x64] + os: [sunos] + requiresBuild: true + dev: true + optional: true + + /@esbuild/win32-arm64@0.17.19: + resolution: {integrity: sha512-yYx+8jwowUstVdorcMdNlzklLYhPxjniHWFKgRqH7IFlUEa0Umu3KuYplf1HUZZ422e3NU9F4LGb+4O0Kdcaag==} + engines: {node: '>=12'} + cpu: [arm64] + os: [win32] + requiresBuild: true + dev: true + optional: true + + /@esbuild/win32-ia32@0.17.19: + resolution: {integrity: sha512-eggDKanJszUtCdlVs0RB+h35wNlb5v4TWEkq4vZcmVt5u/HiDZrTXe2bWFQUez3RgNHwx/x4sk5++4NSSicKkw==} + engines: {node: '>=12'} + cpu: [ia32] + os: [win32] + requiresBuild: true + dev: true + optional: true + + /@esbuild/win32-x64@0.17.19: + resolution: {integrity: sha512-lAhycmKnVOuRYNtRtatQR1LPQf2oYCkRGkSFnseDAKPl8lu5SOsK/e1sXe5a0Pc5kHIHe6P2I/ilntNv2xf3cA==} + engines: {node: '>=12'} + cpu: [x64] + os: [win32] + requiresBuild: true + dev: true + optional: true + + /esbuild@0.17.19: + resolution: {integrity: sha512-XQ0jAPFkK/u3LcVRcvVHQcTIqD6E2H1fvZMA5dQPSOWb3suUbWbfbRf94pjc0bNzRYLfIrDRQXr7X+LHIm5oHw==} + engines: {node: '>=12'} + hasBin: true + requiresBuild: true + optionalDependencies: + '@esbuild/android-arm': 0.17.19 + '@esbuild/android-arm64': 0.17.19 + '@esbuild/android-x64': 0.17.19 + '@esbuild/darwin-arm64': 0.17.19 + '@esbuild/darwin-x64': 0.17.19 + '@esbuild/freebsd-arm64': 0.17.19 + '@esbuild/freebsd-x64': 0.17.19 + '@esbuild/linux-arm': 0.17.19 + '@esbuild/linux-arm64': 0.17.19 + '@esbuild/linux-ia32': 0.17.19 + '@esbuild/linux-loong64': 0.17.19 + '@esbuild/linux-mips64el': 0.17.19 + '@esbuild/linux-ppc64': 0.17.19 + '@esbuild/linux-riscv64': 0.17.19 + '@esbuild/linux-s390x': 0.17.19 + '@esbuild/linux-x64': 0.17.19 + '@esbuild/netbsd-x64': 0.17.19 + '@esbuild/openbsd-x64': 0.17.19 + '@esbuild/sunos-x64': 0.17.19 + '@esbuild/win32-arm64': 0.17.19 + '@esbuild/win32-ia32': 0.17.19 + '@esbuild/win32-x64': 0.17.19 + dev: true diff --git a/src/index.js b/src/index.js index 0d1f0051..a8ff3a08 100644 --- a/src/index.js +++ b/src/index.js @@ -1,5 +1,352 @@ const Obsidian = require("obsidian"); -const VecLite = require("vec-lite"); +const VecLite = class { + constructor(config) { + this.config = { + file_name: "embeddings-3.json", + folder_path: ".vec_lite", + exists_adapter: null, + mkdir_adapter: null, + read_adapter: null, + rename_adapter: null, + stat_adapter: null, + write_adapter: null, + ...config, + }; + this.file_name = this.config.file_name; + this.folder_path = config.folder_path; + this.file_path = this.folder_path + "/" + this.file_name; + this.embeddings = false; + } + async file_exists(path) { + if (this.config.exists_adapter) { + return await this.config.exists_adapter(path); + } else { + throw new Error("exists_adapter not set"); + } + } + async mkdir(path) { + if (this.config.mkdir_adapter) { + return await this.config.mkdir_adapter(path); + } else { + throw new Error("mkdir_adapter not set"); + } + } + async read_file(path) { + if (this.config.read_adapter) { + return await this.config.read_adapter(path); + } else { + throw new Error("read_adapter not set"); + } + } + async rename(old_path, new_path) { + if (this.config.rename_adapter) { + return await this.config.rename_adapter(old_path, new_path); + } else { + throw new Error("rename_adapter not set"); + } + } + async stat(path) { + if (this.config.stat_adapter) { + return await this.config.stat_adapter(path); + } else { + throw new Error("stat_adapter not set"); + } + } + async write_file(path, data) { + if (this.config.write_adapter) { + return await this.config.write_adapter(path, data); + } else { + throw new Error("write_adapter not set"); + } + } + async load(retries = 0) { + try { + const embeddings_file = await this.read_file(this.file_path); + this.embeddings = JSON.parse(embeddings_file); + console.log("loaded embeddings file: " + this.file_path); + return true; + } catch (error) { + if (retries < 3) { + console.log("retrying load()"); + await new Promise((r) => setTimeout(r, 1e3 + 1e3 * retries)); + return await this.load(retries + 1); + } + console.log( + "failed to load embeddings file, prompt user to initiate bulk embed" + ); + return false; + } + } + async init_embeddings_file() { + if (!(await this.file_exists(this.folder_path))) { + await this.mkdir(this.folder_path); + console.log("created folder: " + this.folder_path); + } else { + console.log("folder already exists: " + this.folder_path); + } + if (!(await this.file_exists(this.file_path))) { + await this.write_file(this.file_path, "{}"); + console.log("created embeddings file: " + this.file_path); + } else { + console.log("embeddings file already exists: " + this.file_path); + } + } + async save() { + const embeddings = JSON.stringify(this.embeddings); + const embeddings_file_exists = await this.file_exists(this.file_path); + if (embeddings_file_exists) { + const new_file_size = embeddings.length; + const existing_file_size = await this.stat(this.file_path).then( + (stat) => stat.size + ); + if (new_file_size > existing_file_size * 0.5) { + await this.write_file(this.file_path, embeddings); + console.log("embeddings file size: " + new_file_size + " bytes"); + } else { + const warning_message = [ + "Warning: New embeddings file size is significantly smaller than existing embeddings file size.", + "Aborting to prevent possible loss of embeddings data.", + "New file size: " + new_file_size + " bytes.", + "Existing file size: " + existing_file_size + " bytes.", + "Restarting Obsidian may fix this.", + ]; + console.log(warning_message.join(" ")); + await this.write_file( + this.folder_path + "/unsaved-embeddings.json", + embeddings + ); + throw new Error( + "Error: New embeddings file size is significantly smaller than existing embeddings file size. Aborting to prevent possible loss of embeddings data." + ); + } + } else { + await this.init_embeddings_file(); + return await this.save(); + } + return true; + } + cos_sim(vector1, vector2) { + let dotProduct = 0; + let normA = 0; + let normB = 0; + for (let i = 0; i < vector1.length; i++) { + dotProduct += vector1[i] * vector2[i]; + normA += vector1[i] * vector1[i]; + normB += vector2[i] * vector2[i]; + } + if (normA === 0 || normB === 0) { + return 0; + } else { + return dotProduct / (Math.sqrt(normA) * Math.sqrt(normB)); + } + } + nearest(to_vec, filter = {}) { + filter = { + results_count: 30, + ...filter, + }; + let nearest = []; + const from_keys = Object.keys(this.embeddings); + for (let i = 0; i < from_keys.length; i++) { + if (filter.skip_sections) { + const from_path = this.embeddings[from_keys[i]].meta.path; + if (from_path.indexOf("#") > -1) continue; + } + if (filter.skip_key) { + if (filter.skip_key === from_keys[i]) continue; + if (filter.skip_key === this.embeddings[from_keys[i]].meta.parent) + continue; + } + if (filter.path_begins_with) { + if ( + typeof filter.path_begins_with === "string" && + !this.embeddings[from_keys[i]].meta.path.startsWith( + filter.path_begins_with + ) + ) + continue; + if ( + Array.isArray(filter.path_begins_with) && + !filter.path_begins_with.some((path) => + this.embeddings[from_keys[i]].meta.path.startsWith(path) + ) + ) + continue; + } + nearest.push({ + link: this.embeddings[from_keys[i]].meta.path, + similarity: this.cos_sim(to_vec, this.embeddings[from_keys[i]].vec), + size: this.embeddings[from_keys[i]].meta.size, + }); + } + nearest.sort(function (a, b) { + return b.similarity - a.similarity; + }); + nearest = nearest.slice(0, filter.results_count); + return nearest; + } + find_nearest_embeddings(to_vec, filter = {}) { + const default_filter = { + max: this.max_sources, + }; + filter = { ...default_filter, ...filter }; + if (Array.isArray(to_vec) && to_vec.length !== this.vec_len) { + this.nearest = {}; + for (let i = 0; i < to_vec.length; i++) { + this.find_nearest_embeddings(to_vec[i], { + max: Math.floor(filter.max / to_vec.length), + }); + } + } else { + const from_keys = Object.keys(this.embeddings); + for (let i = 0; i < from_keys.length; i++) { + if (this.validate_type(this.embeddings[from_keys[i]])) continue; + const sim = this.computeCosineSimilarity( + to_vec, + this.embeddings[from_keys[i]].vec + ); + if (this.nearest[from_keys[i]]) { + this.nearest[from_keys[i]] += sim; + } else { + this.nearest[from_keys[i]] = sim; + } + } + } + let nearest = Object.keys(this.nearest).map((key) => { + return { + key, + similarity: this.nearest[key], + }; + }); + nearest = this.sort_by_similarity(nearest); + nearest = nearest.slice(0, filter.max); + nearest = nearest.map((item) => { + return { + link: this.embeddings[item.key].meta.path, + similarity: item.similarity, + len: + this.embeddings[item.key].meta.len || + this.embeddings[item.key].meta.size, + }; + }); + return nearest; + } + sort_by_similarity(nearest) { + return nearest.sort(function (a, b) { + const a_score = a.similarity; + const b_score = b.similarity; + if (a_score > b_score) return -1; + if (a_score < b_score) return 1; + return 0; + }); + } + // check if key from embeddings exists in files + clean_up_embeddings(files) { + console.log("cleaning up embeddings"); + const keys = Object.keys(this.embeddings); + let deleted_embeddings = 0; + for (const key of keys) { + const path = this.embeddings[key].meta.path; + if (!files.find((file) => path.startsWith(file.path))) { + delete this.embeddings[key]; + deleted_embeddings++; + continue; + } + if (path.indexOf("#") > -1) { + const parent_key = this.embeddings[key].meta.parent; + if (!this.embeddings[parent_key]) { + delete this.embeddings[key]; + deleted_embeddings++; + continue; + } + if (!this.embeddings[parent_key].meta) { + delete this.embeddings[key]; + deleted_embeddings++; + continue; + } + if ( + this.embeddings[parent_key].meta.children && + this.embeddings[parent_key].meta.children.indexOf(key) < 0 + ) { + delete this.embeddings[key]; + deleted_embeddings++; + continue; + } + } + } + return { deleted_embeddings, total_embeddings: keys.length }; + } + get(key) { + return this.embeddings[key] || null; + } + get_meta(key) { + const embedding = this.get(key); + if (embedding && embedding.meta) { + return embedding.meta; + } + return null; + } + get_mtime(key) { + const meta = this.get_meta(key); + if (meta && meta.mtime) { + return meta.mtime; + } + return null; + } + get_hash(key) { + const meta = this.get_meta(key); + if (meta && meta.hash) { + return meta.hash; + } + return null; + } + get_size(key) { + const meta = this.get_meta(key); + if (meta && meta.size) { + return meta.size; + } + return null; + } + get_children(key) { + const meta = this.get_meta(key); + if (meta && meta.children) { + return meta.children; + } + return null; + } + get_vec(key) { + const embedding = this.get(key); + if (embedding && embedding.vec) { + return embedding.vec; + } + return null; + } + save_embedding(key, vec, meta) { + this.embeddings[key] = { + vec, + meta, + }; + } + mtime_is_current(key, source_mtime) { + const mtime = this.get_mtime(key); + if (mtime && mtime >= source_mtime) { + return true; + } + return false; + } + async force_refresh() { + this.embeddings = null; + this.embeddings = {}; + let current_datetime = Math.floor(Date.now() / 1e3); + await this.rename( + this.file_path, + this.folder_path + "/embeddings-" + current_datetime + ".json" + ); + await this.init_embeddings_file(); + } +}; + + const DEFAULT_SETTINGS = { api_key: "", @@ -247,36 +594,42 @@ class SmartConnectionsPlugin extends Obsidian.Plugin { // load file exclusions if not blank if(this.settings.file_exclusions && this.settings.file_exclusions.length > 0) { // split file exclusions into array and trim whitespace - this.file_exclusions = this.settings.file_exclusions.split(",").map((file) => { + this.file_exclusions = this.settings.file_exclusions.split(/[,\n\s]/) + .filter((file) => file.length > 0) + .map((file) => { return file.trim(); }); } // load folder exclusions if not blank if(this.settings.folder_exclusions && this.settings.folder_exclusions.length > 0) { // add slash to end of folder name if not present - const folder_exclusions = this.settings.folder_exclusions.split(",").map((folder) => { - // trim whitespace - folder = folder.trim(); - if(folder.slice(-1) !== "/") { - return folder + "/"; - } else { - return folder; - } - }); + const folder_exclusions = this.settings.folder_exclusions + .split(/[,\n\s]/) + .filter((file) => file.length > 0) + .map((folder) => { + folder = folder.trim(); + return folder.slice(-1) === "/" ? folder : `${folder}/`; + }); // merge folder exclusions with file exclusions this.file_exclusions = this.file_exclusions.concat(folder_exclusions); } // load header exclusions if not blank if(this.settings.header_exclusions && this.settings.header_exclusions.length > 0) { - this.header_exclusions = this.settings.header_exclusions.split(",").map((header) => { - return header.trim(); - }); + this.header_exclusions = this.settings.header_exclusions + .split(/[\s\n,]/) + .filter((file) => file.length > 0) + .map((header) => { + return header.trim(); + }); } // load path_only if not blank if(this.settings.path_only && this.settings.path_only.length > 0) { - this.path_only = this.settings.path_only.split(",").map((path) => { - return path.trim(); - }); + this.path_only = this.settings.path_only + .split(/[\s\n,]/) + .filter((file) => file.length > 0) + .map((path) => { + return path.trim(); + }); } // load self_ref_kw_regex this.self_ref_kw_regex = new RegExp(`\\b(${SMART_TRANSLATION[this.settings.language].pronous.join("|")})\\b`, "gi"); @@ -475,10 +828,10 @@ class SmartConnectionsPlugin extends Obsidian.Plugin { } // skip files where path contains any exclusions let skip = false; - for(let j = 0; j < this.file_exclusions.length; j++) { - if(files[i].path.indexOf(this.file_exclusions[j]) > -1) { + for (const fileExclusion of this.file_exclusions) { + if(files[i].path.includes(fileExclusion)) { skip = true; - this.log_exclusion(this.file_exclusions[j]); + this.log_exclusion(fileExclusion); // break out of loop break; } @@ -1479,6 +1832,7 @@ class SmartConnectionsPlugin extends Obsidian.Plugin { * BEGIN EXTERNAL LINK LOGIC * if link is an object, it indicates external link */ + console.log(this); if (typeof nearest[i].link === "object") { const item = list.createEl("div", { cls: "search-result" }); const link = item.createEl("a", { @@ -2281,7 +2635,7 @@ class SmartConnectionsViewApi { this.plugin = plugin; this.view = view; } - async search (search_text) { + async search(search_text) { return await this.plugin.api.search(search_text); } // trigger reload of embeddings file @@ -2289,6 +2643,23 @@ class SmartConnectionsViewApi { await this.plugin.init_vecs(); await this.view.render_connections(); } + async init_vecs() { + this.smart_vec_lite = new VecLite({ + folder_path: ".smart-connections", + exists_adapter: this.app.vault.adapter.exists.bind( + this.app.vault.adapter + ), + mkdir_adapter: this.app.vault.adapter.mkdir.bind(this.app.vault.adapter), + read_adapter: this.app.vault.adapter.read.bind(this.app.vault.adapter), + rename_adapter: this.app.vault.adapter.rename.bind( + this.app.vault.adapter + ), + stat_adapter: this.app.vault.adapter.stat.bind(this.app.vault.adapter), + write_adapter: this.app.vault.adapter.write.bind(this.app.vault.adapter), + }); + this.embeddings_loaded = await this.smart_vec_lite.load(); + return this.embeddings_loaded; + } } class ScSearchApi { constructor(app, plugin) { @@ -3592,8 +3963,6 @@ class SmartConnectionsChatModel { // if contains dataview code block get all dataview code blocks file_content = await this.render_dataview_queries(file_content, note.path, opts); } - // cut off note if %%---%% is found - file_content = file_content.replace(/%%---%%[\s\S]*/, ""); return file_content.substring(0, opts.char_limit); } diff --git a/src/vec_lite.js b/src/vec_lite.js new file mode 100644 index 00000000..cc301f79 --- /dev/null +++ b/src/vec_lite.js @@ -0,0 +1,347 @@ +export default class { + constructor(config) { + this.config = { + file_name: "embeddings-3.json", + folder_path: ".vec_lite", + exists_adapter: null, + mkdir_adapter: null, + read_adapter: null, + rename_adapter: null, + stat_adapter: null, + write_adapter: null, + ...config, + }; + this.file_name = this.config.file_name; + this.folder_path = config.folder_path; + this.file_path = this.folder_path + "/" + this.file_name; + this.embeddings = false; + } + async file_exists(path) { + if (this.config.exists_adapter) { + return await this.config.exists_adapter(path); + } else { + throw new Error("exists_adapter not set"); + } + } + async mkdir(path) { + if (this.config.mkdir_adapter) { + return await this.config.mkdir_adapter(path); + } else { + throw new Error("mkdir_adapter not set"); + } + } + async read_file(path) { + if (this.config.read_adapter) { + return await this.config.read_adapter(path); + } else { + throw new Error("read_adapter not set"); + } + } + async rename(old_path, new_path) { + if (this.config.rename_adapter) { + return await this.config.rename_adapter(old_path, new_path); + } else { + throw new Error("rename_adapter not set"); + } + } + async stat(path) { + if (this.config.stat_adapter) { + return await this.config.stat_adapter(path); + } else { + throw new Error("stat_adapter not set"); + } + } + async write_file(path, data) { + if (this.config.write_adapter) { + return await this.config.write_adapter(path, data); + } else { + throw new Error("write_adapter not set"); + } + } + async load(retries = 0) { + try { + const embeddings_file = await this.read_file(this.file_path); + this.embeddings = JSON.parse(embeddings_file); + console.log("loaded embeddings file: " + this.file_path); + return true; + } catch (error) { + if (retries < 3) { + console.log("retrying load()"); + await new Promise((r) => setTimeout(r, 1e3 + 1e3 * retries)); + return await this.load(retries + 1); + } + console.log( + "failed to load embeddings file, prompt user to initiate bulk embed" + ); + return false; + } + } + async init_embeddings_file() { + if (!(await this.file_exists(this.folder_path))) { + await this.mkdir(this.folder_path); + console.log("created folder: " + this.folder_path); + } else { + console.log("folder already exists: " + this.folder_path); + } + if (!(await this.file_exists(this.file_path))) { + await this.write_file(this.file_path, "{}"); + console.log("created embeddings file: " + this.file_path); + } else { + console.log("embeddings file already exists: " + this.file_path); + } + } + async save() { + const embeddings = JSON.stringify(this.embeddings); + const embeddings_file_exists = await this.file_exists(this.file_path); + if (embeddings_file_exists) { + const new_file_size = embeddings.length; + const existing_file_size = await this.stat(this.file_path).then( + (stat) => stat.size + ); + if (new_file_size > existing_file_size * 0.5) { + await this.write_file(this.file_path, embeddings); + console.log("embeddings file size: " + new_file_size + " bytes"); + } else { + const warning_message = [ + "Warning: New embeddings file size is significantly smaller than existing embeddings file size.", + "Aborting to prevent possible loss of embeddings data.", + "New file size: " + new_file_size + " bytes.", + "Existing file size: " + existing_file_size + " bytes.", + "Restarting Obsidian may fix this.", + ]; + console.log(warning_message.join(" ")); + await this.write_file( + this.folder_path + "/unsaved-embeddings.json", + embeddings + ); + throw new Error( + "Error: New embeddings file size is significantly smaller than existing embeddings file size. Aborting to prevent possible loss of embeddings data." + ); + } + } else { + await this.init_embeddings_file(); + return await this.save(); + } + return true; + } + cos_sim(vector1, vector2) { + let dotProduct = 0; + let normA = 0; + let normB = 0; + for (let i = 0; i < vector1.length; i++) { + dotProduct += vector1[i] * vector2[i]; + normA += vector1[i] * vector1[i]; + normB += vector2[i] * vector2[i]; + } + if (normA === 0 || normB === 0) { + return 0; + } else { + return dotProduct / (Math.sqrt(normA) * Math.sqrt(normB)); + } + } + nearest(to_vec, filter = {}) { + filter = { + results_count: 30, + ...filter, + }; + let nearest = []; + const from_keys = Object.keys(this.embeddings); + for (let i = 0; i < from_keys.length; i++) { + if (filter.skip_sections) { + const from_path = this.embeddings[from_keys[i]].meta.path; + if (from_path.indexOf("#") > -1) continue; + } + if (filter.skip_key) { + if (filter.skip_key === from_keys[i]) continue; + if (filter.skip_key === this.embeddings[from_keys[i]].meta.parent) + continue; + } + if (filter.path_begins_with) { + if ( + typeof filter.path_begins_with === "string" && + !this.embeddings[from_keys[i]].meta.path.startsWith( + filter.path_begins_with + ) + ) + continue; + if ( + Array.isArray(filter.path_begins_with) && + !filter.path_begins_with.some((path) => + this.embeddings[from_keys[i]].meta.path.startsWith(path) + ) + ) + continue; + } + nearest.push({ + link: this.embeddings[from_keys[i]].meta.path, + similarity: this.cos_sim(to_vec, this.embeddings[from_keys[i]].vec), + size: this.embeddings[from_keys[i]].meta.size, + }); + } + nearest.sort(function (a, b) { + return b.similarity - a.similarity; + }); + nearest = nearest.slice(0, filter.results_count); + return nearest; + } + find_nearest_embeddings(to_vec, filter = {}) { + const default_filter = { + max: this.max_sources, + }; + filter = { ...default_filter, ...filter }; + if (Array.isArray(to_vec) && to_vec.length !== this.vec_len) { + this.nearest = {}; + for (let i = 0; i < to_vec.length; i++) { + this.find_nearest_embeddings(to_vec[i], { + max: Math.floor(filter.max / to_vec.length), + }); + } + } else { + const from_keys = Object.keys(this.embeddings); + for (let i = 0; i < from_keys.length; i++) { + if (this.validate_type(this.embeddings[from_keys[i]])) continue; + const sim = this.computeCosineSimilarity( + to_vec, + this.embeddings[from_keys[i]].vec + ); + if (this.nearest[from_keys[i]]) { + this.nearest[from_keys[i]] += sim; + } else { + this.nearest[from_keys[i]] = sim; + } + } + } + let nearest = Object.keys(this.nearest).map((key) => { + return { + key, + similarity: this.nearest[key], + }; + }); + nearest = this.sort_by_similarity(nearest); + nearest = nearest.slice(0, filter.max); + nearest = nearest.map((item) => { + return { + link: this.embeddings[item.key].meta.path, + similarity: item.similarity, + len: + this.embeddings[item.key].meta.len || + this.embeddings[item.key].meta.size, + }; + }); + return nearest; + } + sort_by_similarity(nearest) { + return nearest.sort(function (a, b) { + const a_score = a.similarity; + const b_score = b.similarity; + if (a_score > b_score) return -1; + if (a_score < b_score) return 1; + return 0; + }); + } + // check if key from embeddings exists in files + clean_up_embeddings(files) { + console.log("cleaning up embeddings"); + const keys = Object.keys(this.embeddings); + let deleted_embeddings = 0; + for (const key of keys) { + const path = this.embeddings[key].meta.path; + if (!files.find((file) => path.startsWith(file.path))) { + delete this.embeddings[key]; + deleted_embeddings++; + continue; + } + if (path.indexOf("#") > -1) { + const parent_key = this.embeddings[key].meta.parent; + if (!this.embeddings[parent_key]) { + delete this.embeddings[key]; + deleted_embeddings++; + continue; + } + if (!this.embeddings[parent_key].meta) { + delete this.embeddings[key]; + deleted_embeddings++; + continue; + } + if ( + this.embeddings[parent_key].meta.children && + this.embeddings[parent_key].meta.children.indexOf(key) < 0 + ) { + delete this.embeddings[key]; + deleted_embeddings++; + continue; + } + } + } + return { deleted_embeddings, total_embeddings: keys.length }; + } + get(key) { + return this.embeddings[key] || null; + } + get_meta(key) { + const embedding = this.get(key); + if (embedding && embedding.meta) { + return embedding.meta; + } + return null; + } + get_mtime(key) { + const meta = this.get_meta(key); + if (meta && meta.mtime) { + return meta.mtime; + } + return null; + } + get_hash(key) { + const meta = this.get_meta(key); + if (meta && meta.hash) { + return meta.hash; + } + return null; + } + get_size(key) { + const meta = this.get_meta(key); + if (meta && meta.size) { + return meta.size; + } + return null; + } + get_children(key) { + const meta = this.get_meta(key); + if (meta && meta.children) { + return meta.children; + } + return null; + } + get_vec(key) { + const embedding = this.get(key); + if (embedding && embedding.vec) { + return embedding.vec; + } + return null; + } + save_embedding(key, vec, meta) { + this.embeddings[key] = { + vec, + meta, + }; + } + mtime_is_current(key, source_mtime) { + const mtime = this.get_mtime(key); + if (mtime && mtime >= source_mtime) { + return true; + } + return false; + } + async force_refresh() { + this.embeddings = null; + this.embeddings = {}; + let current_datetime = Math.floor(Date.now() / 1e3); + await this.rename( + this.file_path, + this.folder_path + "/embeddings-" + current_datetime + ".json" + ); + await this.init_embeddings_file(); + } +}; + diff --git a/styles.css b/styles.css new file mode 100644 index 00000000..70a98ce4 --- /dev/null +++ b/styles.css @@ -0,0 +1,206 @@ +.sc-list .tree-item-self { + cursor: pointer; +} +.sc-list .tree-item-self small { + color: var(--color-gray-40); +} +.sc-brand { + position: fixed; + bottom: 0; + right: 0; + background-color: var(--titlebar-background); +} +.sc-brand > svg, .sc-brand > p { + display: inline; + margin: 0 0.1rem 0 0.3rem; + font-size: 0.77rem; + line-height: 1; + color: var(--tab-text-color-focused-active-current); + height: 0.88rem; + width: auto; +} +.sc-brand > svg, .sc-brand > p > a { + color: var(--tab-text-color-focused-active-current); +} +.sc-context { + color: var(--nav-item-color); + font-size: var(--nav-item-size); + margin: 0.5em 0.5em 1em; + width: 100%; +} +.sc-list > .sc-collapsed ul { + display: none; +} +.sc-list > .sc-collapsed span svg { + transform: rotate(-90deg); +} +.sc-list > :not(.sc-collapsed) span svg { + transform: rotate(0deg); +} +.sc-list > div span svg { + height: auto; + margin: auto 0.5em auto 0; + flex: none; +} +.sc-list > div > span { + display: inline-flex; + width: 100%; + padding-left: 0; +} +.sc-list > div ul { + margin: 0; + padding-left: 1.3rem; +} +.sc-list > div > a { + display: block; +} +.sc-list > div > ul > li { + display: block; +} +/* Only on right sidebar */ +.mod-right-split .sc-list .search-result { + font-size: var(--font-text-size); + font-size: 0.88rem; +} +.sc-list .search-result { + cursor: pointer; + background: transparent; + padding: var(--nav-item-padding); + margin-bottom: 1px; + align-items: baseline; + border-radius: var(--radius-s); + font-weight: var(--nav-item-weight); +} +.sc-list .search-result:hover { + color: var(--nav-item-color-active); + background-color: var(--nav-item-background-active); + font-weight: var(--nav-item-weight-active); +} +.sc-list .search-result span { + color: var(--h5-color); +} +.sc-list .search-result small { + color: var(--h5-color); + font-size: 0.8rem; + font-weight: 500; +} +.sc-list .search-result p { + margin-top: 0.3rem; + margin-bottom: 0.3rem; +} +.sc-top-bar { + display: flex; + width: 100%; + justify-content: end; +} +.sc-top-bar .search-input-container { + width: calc(100% - var(--size-4-8)); + margin: 4px auto; +} + +/* Chat */ +.sc-top-bar-container { + align-self: flex-end; + display: flex; + width: 100%; +} +.sc-top-bar-container input.sc-chat-name-input{ + flex-grow: 1; +} +.sc-chat-container { + display: flex; + flex-direction: column; + align-items: center; + justify-content: flex-end; + height: 100%; + padding: 10px; +} +.sc-chat-box { + display: flex; + flex-direction: column; + align-items: flex-start; + justify-content: flex-end; + height: 100%; + width: 100%; + overflow: hidden; + user-select: text; +} +.sc-message-container { + border: 1px solid var(--blockquote-border-color); + border-radius: 10px; + margin: 0.5rem 0; + display: flex; + flex-direction: column; + align-items: flex-start; + width: 100%; + height: 100%; + overflow-y: scroll; +} +.sc-message { + max-width: 90%; + margin: 10px; + padding: 10px; + border-radius: 10px; + word-break: break-word; +} + +.sc-message.assistant { + background-color: #007bff; + color: #fff; +} +.sc-message.user { + background-color: #333; + color: #fff; + align-self: flex-end; +} + +.sc-message-content { + margin: 0; +} +.sc-message-content p { + margin: 10px; +} + +.sc-chat-form { + display: flex; + padding: 0 10px 1rem 0; + width: 100%; + max-height: 50%; +} + +.sc-chat-input { + flex-grow: 1; + padding: 10px; + border: none; + border-radius: 10px; + resize: none; + height: auto; + max-height: 100%; + border: 1px solid var(--blockquote-border-color); +} + +.send-button { + margin-left: 10px; + padding: 10px; + border: none; + border-radius: 10px; + cursor: pointer; +} + +.sc-msg-button { + cursor: pointer; + float: right; + margin-left: 5px; + opacity: 0.8; +} +.sc-msg-button:hover { + opacity: 1; +} +#sc-abort-button { + cursor: pointer; + padding: 10px; + border-radius: 5px; +} +#sc-abort-button:hover { + background-color: var(--nav-item-background-active); +} \ No newline at end of file From 0d4cc3f5f10d596124a3e18e032b2d07a46f4a4f Mon Sep 17 00:00:00 2001 From: Mara Date: Tue, 2 Jan 2024 13:18:49 +0100 Subject: [PATCH 08/11] fusion --- data.json | 4 +- main.js | 3382 -------------------------------------------------- src/index.js | 1 - 3 files changed, 2 insertions(+), 3385 deletions(-) delete mode 100644 main.js diff --git a/data.json b/data.json index f592d2cd..aa5ba945 100644 --- a/data.json +++ b/data.json @@ -1,13 +1,13 @@ { "api_key": "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", - "chat_open": true, + "chat_open": false, "file_exclusions": "sortspec", "folder_exclusions": "", "header_exclusions": "", "path_only": "", "show_full_path": false, "cut_off_frontmatter": true, - "expanded_view": true, + "expanded_view": false, "group_nearest_by_file": false, "language": "fr", "log_render": false, diff --git a/main.js b/main.js deleted file mode 100644 index ecff7aeb..00000000 --- a/main.js +++ /dev/null @@ -1,3382 +0,0 @@ -// src/index.js -var Obsidian = require("obsidian"); -var VecLite = class { - constructor(config) { - this.config = { - file_name: "embeddings-3.json", - folder_path: ".vec_lite", - exists_adapter: null, - mkdir_adapter: null, - read_adapter: null, - rename_adapter: null, - stat_adapter: null, - write_adapter: null, - ...config - }; - this.file_name = this.config.file_name; - this.folder_path = config.folder_path; - this.file_path = this.folder_path + "/" + this.file_name; - this.embeddings = false; - } - async file_exists(path) { - if (this.config.exists_adapter) { - return await this.config.exists_adapter(path); - } else { - throw new Error("exists_adapter not set"); - } - } - async mkdir(path) { - if (this.config.mkdir_adapter) { - return await this.config.mkdir_adapter(path); - } else { - throw new Error("mkdir_adapter not set"); - } - } - async read_file(path) { - if (this.config.read_adapter) { - return await this.config.read_adapter(path); - } else { - throw new Error("read_adapter not set"); - } - } - async rename(old_path, new_path) { - if (this.config.rename_adapter) { - return await this.config.rename_adapter(old_path, new_path); - } else { - throw new Error("rename_adapter not set"); - } - } - async stat(path) { - if (this.config.stat_adapter) { - return await this.config.stat_adapter(path); - } else { - throw new Error("stat_adapter not set"); - } - } - async write_file(path, data) { - if (this.config.write_adapter) { - return await this.config.write_adapter(path, data); - } else { - throw new Error("write_adapter not set"); - } - } - async load(retries = 0) { - try { - const embeddings_file = await this.read_file(this.file_path); - this.embeddings = JSON.parse(embeddings_file); - console.log("loaded embeddings file: " + this.file_path); - return true; - } catch (error) { - if (retries < 3) { - console.log("retrying load()"); - await new Promise((r) => setTimeout(r, 1e3 + 1e3 * retries)); - return await this.load(retries + 1); - } - console.log( - "failed to load embeddings file, prompt user to initiate bulk embed" - ); - return false; - } - } - async init_embeddings_file() { - if (!await this.file_exists(this.folder_path)) { - await this.mkdir(this.folder_path); - console.log("created folder: " + this.folder_path); - } else { - console.log("folder already exists: " + this.folder_path); - } - if (!await this.file_exists(this.file_path)) { - await this.write_file(this.file_path, "{}"); - console.log("created embeddings file: " + this.file_path); - } else { - console.log("embeddings file already exists: " + this.file_path); - } - } - async save() { - const embeddings = JSON.stringify(this.embeddings); - const embeddings_file_exists = await this.file_exists(this.file_path); - if (embeddings_file_exists) { - const new_file_size = embeddings.length; - const existing_file_size = await this.stat(this.file_path).then( - (stat) => stat.size - ); - if (new_file_size > existing_file_size * 0.5) { - await this.write_file(this.file_path, embeddings); - console.log("embeddings file size: " + new_file_size + " bytes"); - } else { - const warning_message = [ - "Warning: New embeddings file size is significantly smaller than existing embeddings file size.", - "Aborting to prevent possible loss of embeddings data.", - "New file size: " + new_file_size + " bytes.", - "Existing file size: " + existing_file_size + " bytes.", - "Restarting Obsidian may fix this." - ]; - console.log(warning_message.join(" ")); - await this.write_file( - this.folder_path + "/unsaved-embeddings.json", - embeddings - ); - throw new Error( - "Error: New embeddings file size is significantly smaller than existing embeddings file size. Aborting to prevent possible loss of embeddings data." - ); - } - } else { - await this.init_embeddings_file(); - return await this.save(); - } - return true; - } - cos_sim(vector1, vector2) { - let dotProduct = 0; - let normA = 0; - let normB = 0; - for (let i = 0; i < vector1.length; i++) { - dotProduct += vector1[i] * vector2[i]; - normA += vector1[i] * vector1[i]; - normB += vector2[i] * vector2[i]; - } - if (normA === 0 || normB === 0) { - return 0; - } else { - return dotProduct / (Math.sqrt(normA) * Math.sqrt(normB)); - } - } - nearest(to_vec, filter = {}) { - filter = { - results_count: 30, - ...filter - }; - let nearest = []; - const from_keys = Object.keys(this.embeddings); - for (let i = 0; i < from_keys.length; i++) { - if (filter.skip_sections) { - const from_path = this.embeddings[from_keys[i]].meta.path; - if (from_path.indexOf("#") > -1) - continue; - } - if (filter.skip_key) { - if (filter.skip_key === from_keys[i]) - continue; - if (filter.skip_key === this.embeddings[from_keys[i]].meta.parent) - continue; - } - if (filter.path_begins_with) { - if (typeof filter.path_begins_with === "string" && !this.embeddings[from_keys[i]].meta.path.startsWith( - filter.path_begins_with - )) - continue; - if (Array.isArray(filter.path_begins_with) && !filter.path_begins_with.some( - (path) => this.embeddings[from_keys[i]].meta.path.startsWith(path) - )) - continue; - } - nearest.push({ - link: this.embeddings[from_keys[i]].meta.path, - similarity: this.cos_sim(to_vec, this.embeddings[from_keys[i]].vec), - size: this.embeddings[from_keys[i]].meta.size - }); - } - nearest.sort(function(a, b) { - return b.similarity - a.similarity; - }); - nearest = nearest.slice(0, filter.results_count); - return nearest; - } - find_nearest_embeddings(to_vec, filter = {}) { - const default_filter = { - max: this.max_sources - }; - filter = { ...default_filter, ...filter }; - if (Array.isArray(to_vec) && to_vec.length !== this.vec_len) { - this.nearest = {}; - for (let i = 0; i < to_vec.length; i++) { - this.find_nearest_embeddings(to_vec[i], { - max: Math.floor(filter.max / to_vec.length) - }); - } - } else { - const from_keys = Object.keys(this.embeddings); - for (let i = 0; i < from_keys.length; i++) { - if (this.validate_type(this.embeddings[from_keys[i]])) - continue; - const sim = this.computeCosineSimilarity( - to_vec, - this.embeddings[from_keys[i]].vec - ); - if (this.nearest[from_keys[i]]) { - this.nearest[from_keys[i]] += sim; - } else { - this.nearest[from_keys[i]] = sim; - } - } - } - let nearest = Object.keys(this.nearest).map((key) => { - return { - key, - similarity: this.nearest[key] - }; - }); - nearest = this.sort_by_similarity(nearest); - nearest = nearest.slice(0, filter.max); - nearest = nearest.map((item) => { - return { - link: this.embeddings[item.key].meta.path, - similarity: item.similarity, - len: this.embeddings[item.key].meta.len || this.embeddings[item.key].meta.size - }; - }); - return nearest; - } - sort_by_similarity(nearest) { - return nearest.sort(function(a, b) { - const a_score = a.similarity; - const b_score = b.similarity; - if (a_score > b_score) - return -1; - if (a_score < b_score) - return 1; - return 0; - }); - } - // check if key from embeddings exists in files - clean_up_embeddings(files) { - console.log("cleaning up embeddings"); - const keys = Object.keys(this.embeddings); - let deleted_embeddings = 0; - for (const key of keys) { - const path = this.embeddings[key].meta.path; - if (!files.find((file) => path.startsWith(file.path))) { - delete this.embeddings[key]; - deleted_embeddings++; - continue; - } - if (path.indexOf("#") > -1) { - const parent_key = this.embeddings[key].meta.parent; - if (!this.embeddings[parent_key]) { - delete this.embeddings[key]; - deleted_embeddings++; - continue; - } - if (!this.embeddings[parent_key].meta) { - delete this.embeddings[key]; - deleted_embeddings++; - continue; - } - if (this.embeddings[parent_key].meta.children && this.embeddings[parent_key].meta.children.indexOf(key) < 0) { - delete this.embeddings[key]; - deleted_embeddings++; - continue; - } - } - } - return { deleted_embeddings, total_embeddings: keys.length }; - } - get(key) { - return this.embeddings[key] || null; - } - get_meta(key) { - const embedding = this.get(key); - if (embedding && embedding.meta) { - return embedding.meta; - } - return null; - } - get_mtime(key) { - const meta = this.get_meta(key); - if (meta && meta.mtime) { - return meta.mtime; - } - return null; - } - get_hash(key) { - const meta = this.get_meta(key); - if (meta && meta.hash) { - return meta.hash; - } - return null; - } - get_size(key) { - const meta = this.get_meta(key); - if (meta && meta.size) { - return meta.size; - } - return null; - } - get_children(key) { - const meta = this.get_meta(key); - if (meta && meta.children) { - return meta.children; - } - return null; - } - get_vec(key) { - const embedding = this.get(key); - if (embedding && embedding.vec) { - return embedding.vec; - } - return null; - } - save_embedding(key, vec, meta) { - this.embeddings[key] = { - vec, - meta - }; - } - mtime_is_current(key, source_mtime) { - const mtime = this.get_mtime(key); - if (mtime && mtime >= source_mtime) { - return true; - } - return false; - } - async force_refresh() { - this.embeddings = null; - this.embeddings = {}; - let current_datetime = Math.floor(Date.now() / 1e3); - await this.rename( - this.file_path, - this.folder_path + "/embeddings-" + current_datetime + ".json" - ); - await this.init_embeddings_file(); - } -}; -var DEFAULT_SETTINGS = { - api_key: "", - chat_open: true, - file_exclusions: "", - folder_exclusions: "", - header_exclusions: "", - path_only: "", - show_full_path: false, - cut_off_frontmatter: false, - expanded_view: true, - group_nearest_by_file: false, - language: "en", - log_render: false, - log_render_files: false, - recently_sent_retry_notice: false, - skip_sections: false, - smart_chat_model: "gpt-3.5-turbo-16k", - view_open: true, - version: "", - open_in_big_view: false -}; -var MAX_EMBED_STRING_LENGTH = 25e3; -var VERSION; -var SUPPORTED_FILE_TYPES = ["md", "canvas"]; -var SMART_TRANSLATION = { - "en": { - "pronous": ["my", "I", "me", "mine", "our", "ours", "us", "we"], - "prompt": "Based on your notes", - "initial_message": "Hi, I'm ChatGPT with access to your notes via Smart Connections. Ask me a question about your notes and I'll try to answer it.", - "try_placeholder": `Try "Based on my notes" or "Summarize [[this note]]" or "Important tasks in /folder/"` - }, - "es": { - "pronous": ["mi", "yo", "m\xED", "t\xFA"], - "prompt": "Bas\xE1ndose en sus notas", - "initial_message": "Hola, soy ChatGPT con acceso a tus apuntes a trav\xE9s de Smart Connections. Hazme una pregunta sobre tus apuntes e intentar\xE9 responderte.", - "try_placeholder": `Prueba "Basado en mis notas" o "Resumen [[esta nota]]" o "Tareas importantes en /carpeta/"` - }, - "fr": { - "pronous": ["me", "mon", "ma", "mes", "moi", "nous", "notre", "nos", "je", "j'", "m'"], - "prompt": "D'apr\xE8s vos notes", - "initial_message": "Bonjour, je suis ChatGPT et j'ai acc\xE8s \xE0 vos notes via Smart Connections. Posez-moi une question sur vos notes et j'essaierai d'y r\xE9pondre.", - "try_placeholder": `Essayez "D'apr\xE8s mes notes" ou "R\xE9sume [[cette note]]" ou "T\xE2ches importantes dans /dossier/"` - }, - "de": { - "pronous": ["mein", "meine", "meinen", "meiner", "meines", "mir", "uns", "unser", "unseren", "unserer", "unseres"], - "prompt": "Basierend auf Ihren Notizen", - "initial_message": "Hallo, ich bin ChatGPT und habe \xFCber Smart Connections Zugang zu Ihren Notizen. Stellen Sie mir eine Frage zu Ihren Notizen und ich werde versuchen, sie zu beantworten.", - "try_placeholder": `Versuchen Sie "Basierend auf meinen Notizen" oder "Zusammenfassen [[dieser Notiz]]" oder "Wichtige Aufgaben im /Ordner/"` - }, - "it": { - "pronous": ["mio", "mia", "miei", "mie", "noi", "nostro", "nostri", "nostra", "nostre"], - "prompt": "Sulla base degli appunti", - "initial_message": "Ciao, sono ChatGPT e ho accesso ai tuoi appunti tramite Smart Connections. Fatemi una domanda sui vostri appunti e cercher\xF2 di rispondervi.", - "try_placeholder": `Prova "Sulla base dei miei appunti" o "Riassumi [[questo appunto]]" o "Compiti importanti in /cartella/"` - } -}; -var crypto = require("crypto"); -function md5(str) { - return crypto.createHash("md5").update(str).digest("hex"); -} -var SmartConnectionsPlugin = class extends Obsidian.Plugin { - // constructor - constructor() { - super(...arguments); - this.api = null; - this.embeddings_loaded = false; - this.file_exclusions = []; - this.folders = []; - this.has_new_embeddings = false; - this.header_exclusions = []; - this.nearest_cache = {}; - this.path_only = []; - this.render_log = {}; - this.render_log.deleted_embeddings = 0; - this.render_log.exclusions_logs = {}; - this.render_log.failed_embeddings = []; - this.render_log.files = []; - this.render_log.new_embeddings = 0; - this.render_log.skipped_low_delta = {}; - this.render_log.token_usage = 0; - this.render_log.tokens_saved_by_cache = 0; - this.retry_notice_timeout = null; - this.save_timeout = null; - this.sc_branding = {}; - this.self_ref_kw_regex = null; - this.update_available = false; - } - async onload() { - this.app.workspace.onLayoutReady(this.initialize.bind(this)); - } - onunload() { - this.output_render_log(); - console.log("unloading plugin"); - this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE); - this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE); - } - async initialize() { - console.log("Loading Smart Connections plugin"); - VERSION = this.manifest.version; - await this.loadSettings(); - setTimeout(this.check_for_update.bind(this), 3e3); - setInterval(this.check_for_update.bind(this), 108e5); - this.addIcon(); - this.addCommand({ - id: "sc-find-notes", - name: "Find: Make Smart Connections", - icon: "pencil_icon", - hotkeys: [], - // editorCallback: async (editor) => { - editorCallback: async (editor) => { - if (editor.somethingSelected()) { - let selected_text = editor.getSelection(); - await this.make_connections(selected_text); - } else { - this.nearest_cache = {}; - await this.make_connections(); - } - } - }); - this.addCommand({ - id: "smart-connections-view", - name: "Open: View Smart Connections", - callback: () => { - this.open_view(); - } - }); - this.addCommand({ - id: "smart-connections-chat", - name: "Open: Smart Chat Conversation", - callback: () => { - this.open_chat(); - } - }); - this.addCommand({ - id: "smart-connections-random", - name: "Open: Random Note from Smart Connections", - callback: () => { - this.open_random_note(); - } - }); - this.addSettingTab(new SmartConnectionsSettingsTab(this.app, this)); - this.registerView(SMART_CONNECTIONS_VIEW_TYPE, (leaf) => new SmartConnectionsView(leaf, this)); - this.registerView(SMART_CONNECTIONS_CHAT_VIEW_TYPE, (leaf) => new SmartConnectionsChatView(leaf, this)); - this.registerMarkdownCodeBlockProcessor("smart-connections", this.render_code_block.bind(this)); - if (this.settings.view_open) { - this.open_view(); - } - if (this.settings.chat_open) { - this.open_chat(); - } - if (this.settings.version !== VERSION) { - this.settings.version = VERSION; - await this.saveSettings(); - this.open_view(); - } - this.add_to_gitignore(); - this.api = new ScSearchApi(this.app, this); - (window["SmartSearchApi"] = this.api) && this.register(() => delete window["SmartSearchApi"]); - } - async init_vecs() { - this.smart_vec_lite = new VecLite({ - folder_path: ".smart-connections", - exists_adapter: this.app.vault.adapter.exists.bind(this.app.vault.adapter), - mkdir_adapter: this.app.vault.adapter.mkdir.bind(this.app.vault.adapter), - read_adapter: this.app.vault.adapter.read.bind(this.app.vault.adapter), - rename_adapter: this.app.vault.adapter.rename.bind(this.app.vault.adapter), - stat_adapter: this.app.vault.adapter.stat.bind(this.app.vault.adapter), - write_adapter: this.app.vault.adapter.write.bind(this.app.vault.adapter) - }); - this.embeddings_loaded = await this.smart_vec_lite.load(); - return this.embeddings_loaded; - } - async update_to_v2() { - if (!this.settings.license_key) - return new Obsidian.Notice("[Smart Connections] Supporter license key required for early access to V2"); - const v2 = await (0, Obsidian.requestUrl)({ - url: "https://sync.smartconnections.app/download_v2", - method: "POST", - headers: { - "Content-Type": "application/json" - }, - body: JSON.stringify({ - license_key: this.settings.license_key - }) - }); - if (v2.status !== 200) - return console.error("Error downloading version 2", v2); - console.log(v2); - await this.app.vault.adapter.write(".obsidian/plugins/smart-connections/main.js", v2.json.main); - await this.app.vault.adapter.write(".obsidian/plugins/smart-connections/manifest.json", v2.json.manifest); - await this.app.vault.adapter.write(".obsidian/plugins/smart-connections/styles.css", v2.json.styles); - window.restart_plugin = async (id) => { - console.log("restarting plugin", id); - await window.app.plugins.disablePlugin(id); - await window.app.plugins.enablePlugin(id); - console.log("plugin restarted", id); - }; - window.restart_plugin(this.manifest.id); - } - async loadSettings() { - this.settings = Object.assign({}, DEFAULT_SETTINGS, await this.loadData()); - if (this.settings.file_exclusions && this.settings.file_exclusions.length > 0) { - this.file_exclusions = this.settings.file_exclusions.split(/[,\n\s]/).filter((file) => file.length > 0).map((file) => { - return file.trim(); - }); - } - if (this.settings.folder_exclusions && this.settings.folder_exclusions.length > 0) { - const folder_exclusions = this.settings.folder_exclusions.split(/[,\n\s]/).filter((file) => file.length > 0).map((folder) => { - folder = folder.trim(); - return folder.slice(-1) === "/" ? folder : `${folder}/`; - }); - this.file_exclusions = this.file_exclusions.concat(folder_exclusions); - } - if (this.settings.header_exclusions && this.settings.header_exclusions.length > 0) { - this.header_exclusions = this.settings.header_exclusions.split(/[\s\n,]/).filter((file) => file.length > 0).map((header) => { - return header.trim(); - }); - } - if (this.settings.path_only && this.settings.path_only.length > 0) { - this.path_only = this.settings.path_only.split(/[\s\n,]/).filter((file) => file.length > 0).map((path) => { - return path.trim(); - }); - } - this.self_ref_kw_regex = new RegExp(`\\b(${SMART_TRANSLATION[this.settings.language].pronous.join("|")})\\b`, "gi"); - await this.load_failed_files(); - } - async saveSettings(rerender = false) { - await this.saveData(this.settings); - await this.loadSettings(); - if (rerender) { - this.nearest_cache = {}; - await this.make_connections(); - } - } - // check for update - async check_for_update() { - try { - const response = await (0, Obsidian.requestUrl)({ - url: "https://api.github.com/repos/brianpetro/obsidian-smart-connections/releases/latest", - method: "GET", - headers: { - "Content-Type": "application/json" - }, - contentType: "application/json" - }); - const latest_release = JSON.parse(response.text).tag_name; - if (latest_release !== VERSION) { - new Obsidian.Notice(`[Smart Connections] A new version is available! (v${latest_release})`); - this.update_available = true; - this.render_brand("all"); - } - } catch (error) { - console.log(error); - } - } - async render_code_block(contents, container, ctx) { - let nearest; - if (contents.trim().length > 0) { - nearest = await this.api.search(contents); - } else { - console.log(ctx); - const file = this.app.vault.getAbstractFileByPath(ctx.sourcePath); - nearest = await this.find_note_connections(file); - } - if (nearest.length) { - this.update_results(container, nearest); - } - } - async make_connections(selected_text = null) { - let view = this.get_view(); - if (!view) { - await this.open_view(); - view = this.get_view(); - } - await view.render_connections(selected_text); - } - addIcon() { - Obsidian.addIcon("smart-connections", ` - - - - - - `); - } - // open random note - async open_random_note() { - const curr_file = this.app.workspace.getActiveFile(); - const curr_key = md5(curr_file.path); - if (typeof this.nearest_cache[curr_key] === "undefined") { - new Obsidian.Notice("[Smart Connections] No Smart Connections found. Open a note to get Smart Connections."); - return; - } - const rand = Math.floor(Math.random() * this.nearest_cache[curr_key].length / 2); - const random_file = this.nearest_cache[curr_key][rand]; - this.open_note(random_file); - } - async open_view() { - if (this.get_view()) { - console.log("Smart Connections view already open"); - return; - } - this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE); - await this.app.workspace.getRightLeaf(false).setViewState({ - type: SMART_CONNECTIONS_VIEW_TYPE, - active: true - }); - this.app.workspace.revealLeaf( - this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE)[0] - ); - } - // source: https://github.com/obsidianmd/obsidian-releases/blob/master/plugin-review.md#avoid-managing-references-to-custom-views - get_view() { - for (let leaf of this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE)) { - if (leaf.view instanceof SmartConnectionsView) { - return leaf.view; - } - } - } - // open chat view - async open_chat(retries = 0) { - if (!this.embeddings_loaded) { - console.log("embeddings not loaded yet"); - if (retries < 3) { - setTimeout(() => { - this.open_chat(retries + 1); - }, 1e3 * (retries + 1)); - return; - } - console.log("embeddings still not loaded, opening smart view"); - this.open_view(); - return; - } - this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE); - if (!this.settings.open_in_big_view) { - await this.app.workspace.getRightLeaf(false).setViewState({ - type: SMART_CONNECTIONS_CHAT_VIEW_TYPE, - active: true - }); - } else { - await this.app.workspace.getLeaf(true).setViewState({ - type: SMART_CONNECTIONS_CHAT_VIEW_TYPE, - active: true - }); - } - this.app.workspace.revealLeaf( - this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE)[0] - ); - } - // get embeddings for all files - async get_all_embeddings() { - const files = (await this.app.vault.getFiles()).filter((file) => file instanceof Obsidian.TFile && (file.extension === "md" || file.extension === "canvas")); - const open_files = this.app.workspace.getLeavesOfType("markdown").map((leaf) => leaf.view.file); - const clean_up_log = this.smart_vec_lite.clean_up_embeddings(files); - if (this.settings.log_render) { - this.render_log.total_files = files.length; - this.render_log.deleted_embeddings = clean_up_log.deleted_embeddings; - this.render_log.total_embeddings = clean_up_log.total_embeddings; - } - let batch_promises = []; - for (let i = 0; i < files.length; i++) { - if (files[i].path.indexOf("#") > -1) { - this.log_exclusion("path contains #"); - continue; - } - if (this.smart_vec_lite.mtime_is_current(md5(files[i].path), files[i].stat.mtime)) { - continue; - } - if (this.settings.failed_files.indexOf(files[i].path) > -1) { - if (this.retry_notice_timeout) { - clearTimeout(this.retry_notice_timeout); - this.retry_notice_timeout = null; - } - if (!this.recently_sent_retry_notice) { - new Obsidian.Notice("Smart Connections: Skipping previously failed file, use button in settings to retry"); - this.recently_sent_retry_notice = true; - setTimeout(() => { - this.recently_sent_retry_notice = false; - }, 6e5); - } - continue; - } - let skip = false; - for (const fileExclusion of this.file_exclusions) { - if (files[i].path.includes(fileExclusion)) { - skip = true; - this.log_exclusion(fileExclusion); - break; - } - } - if (skip) { - continue; - } - if (open_files.indexOf(files[i]) > -1) { - continue; - } - try { - batch_promises.push(this.get_file_embeddings(files[i], false)); - } catch (error) { - console.log(error); - } - if (batch_promises.length > 3) { - await Promise.all(batch_promises); - batch_promises = []; - } - if (i > 0 && i % 100 === 0) { - await this.save_embeddings_to_file(); - } - } - await Promise.all(batch_promises); - await this.save_embeddings_to_file(); - if (this.render_log.failed_embeddings.length > 0) { - await this.save_failed_embeddings(); - } - } - async save_embeddings_to_file(force = false) { - if (!this.has_new_embeddings) { - return; - } - if (!force) { - if (this.save_timeout) { - clearTimeout(this.save_timeout); - this.save_timeout = null; - } - this.save_timeout = setTimeout(() => { - this.save_embeddings_to_file(true); - if (this.save_timeout) { - clearTimeout(this.save_timeout); - this.save_timeout = null; - } - }, 3e4); - console.log("scheduled save"); - return; - } - try { - await this.smart_vec_lite.save(); - this.has_new_embeddings = false; - } catch (error) { - console.log(error); - new Obsidian.Notice("Smart Connections: " + error.message); - } - } - // save failed embeddings to file from render_log.failed_embeddings - async save_failed_embeddings() { - let failed_embeddings = []; - const failed_embeddings_file_exists = await this.app.vault.adapter.exists(".smart-connections/failed-embeddings.txt"); - if (failed_embeddings_file_exists) { - failed_embeddings = await this.app.vault.adapter.read(".smart-connections/failed-embeddings.txt"); - failed_embeddings = failed_embeddings.split("\r\n"); - } - failed_embeddings = failed_embeddings.concat(this.render_log.failed_embeddings); - failed_embeddings = [...new Set(failed_embeddings)]; - failed_embeddings.sort(); - failed_embeddings = failed_embeddings.join("\r\n"); - await this.app.vault.adapter.write(".smart-connections/failed-embeddings.txt", failed_embeddings); - await this.load_failed_files(); - } - // load failed files from failed-embeddings.txt - async load_failed_files() { - const failed_embeddings_file_exists = await this.app.vault.adapter.exists(".smart-connections/failed-embeddings.txt"); - if (!failed_embeddings_file_exists) { - this.settings.failed_files = []; - console.log("No failed files."); - return; - } - const failed_embeddings = await this.app.vault.adapter.read(".smart-connections/failed-embeddings.txt"); - const failed_embeddings_array = failed_embeddings.split("\r\n"); - const failed_files = failed_embeddings_array.map((embedding) => embedding.split("#")[0]).reduce((unique, item) => unique.includes(item) ? unique : [...unique, item], []); - this.settings.failed_files = failed_files; - } - // retry failed embeddings - async retry_failed_files() { - this.settings.failed_files = []; - const failed_embeddings_file_exists = await this.app.vault.adapter.exists(".smart-connections/failed-embeddings.txt"); - if (failed_embeddings_file_exists) { - await this.app.vault.adapter.remove(".smart-connections/failed-embeddings.txt"); - } - await this.get_all_embeddings(); - } - // add .smart-connections to .gitignore to prevent issues with large, frequently updated embeddings file(s) - async add_to_gitignore() { - if (!await this.app.vault.adapter.exists(".gitignore")) { - return; - } - let gitignore_file = await this.app.vault.adapter.read(".gitignore"); - if (gitignore_file.indexOf(".smart-connections") < 0) { - let add_to_gitignore = "\n\n# Ignore Smart Connections folder because embeddings file is large and updated frequently"; - add_to_gitignore += "\n.smart-connections"; - await this.app.vault.adapter.write(".gitignore", gitignore_file + add_to_gitignore); - console.log("added .smart-connections to .gitignore"); - } - } - // force refresh embeddings file but first rename existing embeddings file to .smart-connections/embeddings-YYYY-MM-DD.json - async force_refresh_embeddings_file() { - new Obsidian.Notice("Smart Connections: embeddings file Force Refreshed, making new connections..."); - await this.smart_vec_lite.force_refresh(); - await this.get_all_embeddings(); - this.output_render_log(); - new Obsidian.Notice("Smart Connections: embeddings file Force Refreshed, new connections made."); - } - // get embeddings for embed_input - async get_file_embeddings(curr_file, save = true) { - let req_batch = []; - let blocks = []; - const curr_file_key = md5(curr_file.path); - let file_embed_input = curr_file.path.replace(".md", ""); - file_embed_input = file_embed_input.replace(/\//g, " > "); - let path_only = false; - for (let j = 0; j < this.path_only.length; j++) { - if (curr_file.path.indexOf(this.path_only[j]) > -1) { - path_only = true; - console.log("title only file with matcher: " + this.path_only[j]); - break; - } - } - if (path_only) { - req_batch.push([curr_file_key, file_embed_input, { - mtime: curr_file.stat.mtime, - path: curr_file.path - }]); - await this.get_embeddings_batch(req_batch); - return; - } - if (curr_file.extension === "canvas") { - const canvas_contents = await this.app.vault.cachedRead(curr_file); - if (typeof canvas_contents === "string" && canvas_contents.indexOf("nodes") > -1) { - const canvas_json = JSON.parse(canvas_contents); - for (let j = 0; j < canvas_json.nodes.length; j++) { - if (canvas_json.nodes[j].text) { - file_embed_input += "\n" + canvas_json.nodes[j].text; - } - if (canvas_json.nodes[j].file) { - file_embed_input += "\nLink: " + canvas_json.nodes[j].file; - } - } - } - req_batch.push([curr_file_key, file_embed_input, { - mtime: curr_file.stat.mtime, - path: curr_file.path - }]); - await this.get_embeddings_batch(req_batch); - return; - } - const note_contents = await this.app.vault.cachedRead(curr_file); - let processed_since_last_save = 0; - const note_sections = this.block_parser(note_contents, curr_file.path); - if (note_sections.length > 1) { - for (let j = 0; j < note_sections.length; j++) { - const block_embed_input = note_sections[j].text; - const block_key = md5(note_sections[j].path); - blocks.push(block_key); - if (this.smart_vec_lite.get_size(block_key) === block_embed_input.length) { - continue; - } - if (this.smart_vec_lite.mtime_is_current(block_key, curr_file.stat.mtime)) { - continue; - } - const block_hash = md5(block_embed_input.trim()); - if (this.smart_vec_lite.get_hash(block_key) === block_hash) { - continue; - } - req_batch.push([block_key, block_embed_input, { - // oldmtime: curr_file.stat.mtime, - // get current datetime as unix timestamp - mtime: Date.now(), - hash: block_hash, - parent: curr_file_key, - path: note_sections[j].path, - size: block_embed_input.length - }]); - if (req_batch.length > 9) { - await this.get_embeddings_batch(req_batch); - processed_since_last_save += req_batch.length; - if (processed_since_last_save >= 30) { - await this.save_embeddings_to_file(); - processed_since_last_save = 0; - } - req_batch = []; - } - } - } - if (req_batch.length > 0) { - await this.get_embeddings_batch(req_batch); - req_batch = []; - processed_since_last_save += req_batch.length; - } - file_embed_input += `: -`; - if (note_contents.length < MAX_EMBED_STRING_LENGTH) { - file_embed_input += note_contents; - } else { - const note_meta_cache = this.app.metadataCache.getFileCache(curr_file); - if (typeof note_meta_cache.headings === "undefined") { - file_embed_input += note_contents.substring(0, MAX_EMBED_STRING_LENGTH); - } else { - let note_headings = ""; - for (let j = 0; j < note_meta_cache.headings.length; j++) { - const heading_level = note_meta_cache.headings[j].level; - const heading_text = note_meta_cache.headings[j].heading; - let md_heading = ""; - for (let k = 0; k < heading_level; k++) { - md_heading += "#"; - } - note_headings += `${md_heading} ${heading_text} -`; - } - file_embed_input += note_headings; - if (file_embed_input.length > MAX_EMBED_STRING_LENGTH) { - file_embed_input = file_embed_input.substring(0, MAX_EMBED_STRING_LENGTH); - } - } - } - const file_hash = md5(file_embed_input.trim()); - const existing_hash = this.smart_vec_lite.get_hash(curr_file_key); - if (existing_hash && file_hash === existing_hash) { - this.update_render_log(blocks, file_embed_input); - return; - } - ; - const existing_blocks = this.smart_vec_lite.get_children(curr_file_key); - let existing_has_all_blocks = true; - if (existing_blocks && Array.isArray(existing_blocks) && blocks.length > 0) { - for (let j = 0; j < blocks.length; j++) { - if (existing_blocks.indexOf(blocks[j]) === -1) { - existing_has_all_blocks = false; - break; - } - } - } - if (existing_has_all_blocks) { - const curr_file_size = curr_file.stat.size; - const prev_file_size = this.smart_vec_lite.get_size(curr_file_key); - if (prev_file_size) { - const file_delta_pct = Math.round(Math.abs(curr_file_size - prev_file_size) / curr_file_size * 100); - if (file_delta_pct < 10) { - this.render_log.skipped_low_delta[curr_file.name] = file_delta_pct + "%"; - this.update_render_log(blocks, file_embed_input); - return; - } - } - } - let meta = { - mtime: curr_file.stat.mtime, - hash: file_hash, - path: curr_file.path, - size: curr_file.stat.size, - children: blocks - }; - req_batch.push([curr_file_key, file_embed_input, meta]); - await this.get_embeddings_batch(req_batch); - if (save) { - await this.save_embeddings_to_file(); - } - } - update_render_log(blocks, file_embed_input) { - if (blocks.length > 0) { - this.render_log.tokens_saved_by_cache += file_embed_input.length / 2; - } else { - this.render_log.tokens_saved_by_cache += file_embed_input.length / 4; - } - } - async get_embeddings_batch(req_batch) { - console.log("get_embeddings_batch"); - if (req_batch.length === 0) - return; - const embed_inputs = req_batch.map((req) => req[1]); - const requestResults = await this.request_embedding_from_input(embed_inputs); - if (!requestResults) { - console.log("failed embedding batch"); - this.render_log.failed_embeddings = [...this.render_log.failed_embeddings, ...req_batch.map((req) => req[2].path)]; - return; - } - if (requestResults) { - this.has_new_embeddings = true; - if (this.settings.log_render) { - if (this.settings.log_render_files) { - this.render_log.files = [...this.render_log.files, ...req_batch.map((req) => req[2].path)]; - } - this.render_log.new_embeddings += req_batch.length; - this.render_log.token_usage += requestResults.usage.total_tokens; - } - for (let i = 0; i < requestResults.data.length; i++) { - const vec = requestResults.data[i].embedding; - const index = requestResults.data[i].index; - if (vec) { - const key = req_batch[index][0]; - const meta = req_batch[index][2]; - this.smart_vec_lite.save_embedding(key, vec, meta); - } - } - } - } - async request_embedding_from_input(embed_input, retries = 0) { - if (embed_input.length === 0) { - console.log("embed_input is empty"); - return null; - } - const usedParams = { - model: "text-embedding-ada-002", - input: embed_input - }; - const reqParams = { - url: `https://api.openai.com/v1/embeddings`, - method: "POST", - body: JSON.stringify(usedParams), - headers: { - "Content-Type": "application/json", - "Authorization": `Bearer ${this.settings.api_key}` - } - }; - let resp; - try { - resp = await (0, Obsidian.request)(reqParams); - return JSON.parse(resp); - } catch (error) { - if (error.status === 429 && retries < 3) { - retries++; - const backoff = Math.pow(retries, 2); - console.log(`retrying request (429) in ${backoff} seconds...`); - await new Promise((r) => setTimeout(r, 1e3 * backoff)); - return await this.request_embedding_from_input(embed_input, retries); - } - console.log(resp); - console.log(error); - return null; - } - } - async test_api_key() { - const embed_input = "This is a test of the OpenAI API."; - const resp = await this.request_embedding_from_input(embed_input); - if (resp && resp.usage) { - console.log("API key is valid"); - return true; - } else { - console.log("API key is invalid"); - return false; - } - } - output_render_log() { - if (this.settings.log_render) { - if (this.render_log.new_embeddings === 0) { - return; - } else { - console.log(JSON.stringify(this.render_log, null, 2)); - } - } - this.render_log = {}; - this.render_log.deleted_embeddings = 0; - this.render_log.exclusions_logs = {}; - this.render_log.failed_embeddings = []; - this.render_log.files = []; - this.render_log.new_embeddings = 0; - this.render_log.skipped_low_delta = {}; - this.render_log.token_usage = 0; - this.render_log.tokens_saved_by_cache = 0; - } - // find connections by most similar to current note by cosine similarity - async find_note_connections(current_note = null) { - const curr_key = md5(current_note.path); - let nearest = []; - if (this.nearest_cache[curr_key]) { - nearest = this.nearest_cache[curr_key]; - } else { - for (let j = 0; j < this.file_exclusions.length; j++) { - if (current_note.path.indexOf(this.file_exclusions[j]) > -1) { - this.log_exclusion(this.file_exclusions[j]); - return "excluded"; - } - } - setTimeout(() => { - this.get_all_embeddings(); - }, 3e3); - if (this.smart_vec_lite.mtime_is_current(curr_key, current_note.stat.mtime)) { - } else { - await this.get_file_embeddings(current_note); - } - const vec = this.smart_vec_lite.get_vec(curr_key); - if (!vec) { - return "Error getting embeddings for: " + current_note.path; - } - nearest = this.smart_vec_lite.nearest(vec, { - skip_key: curr_key, - skip_sections: this.settings.skip_sections - }); - this.nearest_cache[curr_key] = nearest; - } - return nearest; - } - // create render_log object of exlusions with number of times skipped as value - log_exclusion(exclusion) { - this.render_log.exclusions_logs[exclusion] = (this.render_log.exclusions_logs[exclusion] || 0) + 1; - } - block_parser(markdown, file_path) { - if (this.settings.skip_sections) { - return []; - } - const lines = markdown.split("\n"); - let blocks = []; - let currentHeaders = []; - const file_breadcrumbs = file_path.replace(".md", "").replace(/\//g, " > "); - let block = ""; - let block_headings = ""; - let block_path = file_path; - let last_heading_line = 0; - let i = 0; - let block_headings_list = []; - for (i = 0; i < lines.length; i++) { - const line = lines[i]; - if (!line.startsWith("#") || ["#", " "].indexOf(line[1]) < 0) { - if (line === "") - continue; - if (["- ", "- [ ] "].indexOf(line) > -1) - continue; - if (currentHeaders.length === 0) - continue; - block += "\n" + line; - continue; - } - last_heading_line = i; - if (i > 0 && last_heading_line !== i - 1 && block.indexOf("\n") > -1 && this.validate_headings(block_headings)) { - output_block(); - } - const level = line.split("#").length - 1; - currentHeaders = currentHeaders.filter((header) => header.level < level); - currentHeaders.push({ header: line.replace(/#/g, "").trim(), level }); - block = file_breadcrumbs; - block += ": " + currentHeaders.map((header) => header.header).join(" > "); - block_headings = "#" + currentHeaders.map((header) => header.header).join("#"); - if (block_headings_list.indexOf(block_headings) > -1) { - let count = 1; - while (block_headings_list.indexOf(`${block_headings}{${count}}`) > -1) { - count++; - } - block_headings = `${block_headings}{${count}}`; - } - block_headings_list.push(block_headings); - block_path = file_path + block_headings; - } - if (last_heading_line !== i - 1 && block.indexOf("\n") > -1 && this.validate_headings(block_headings)) - output_block(); - blocks = blocks.filter((b) => b.length > 50); - return blocks; - function output_block() { - const breadcrumbs_length = block.indexOf("\n") + 1; - const block_length = block.length - breadcrumbs_length; - if (block.length > MAX_EMBED_STRING_LENGTH) { - block = block.substring(0, MAX_EMBED_STRING_LENGTH); - } - blocks.push({ text: block.trim(), path: block_path, length: block_length }); - } - } - // reverse-retrieve block given path - async block_retriever(path, limits = {}) { - limits = { - lines: null, - chars_per_line: null, - max_chars: null, - ...limits - }; - if (path.indexOf("#") < 0) { - console.log("not a block path: " + path); - return false; - } - let block = []; - let block_headings = path.split("#").slice(1); - let heading_occurrence = 0; - if (block_headings[block_headings.length - 1].indexOf("{") > -1) { - heading_occurrence = parseInt(block_headings[block_headings.length - 1].split("{")[1].replace("}", "")); - block_headings[block_headings.length - 1] = block_headings[block_headings.length - 1].split("{")[0]; - } - let currentHeaders = []; - let occurrence_count = 0; - let begin_line = 0; - let i = 0; - const file_path = path.split("#")[0]; - const file = this.app.vault.getAbstractFileByPath(file_path); - if (!(file instanceof Obsidian.TFile)) { - console.log("not a file: " + file_path); - return false; - } - const file_contents = await this.app.vault.cachedRead(file); - const lines = file_contents.split("\n"); - let is_code = false; - for (i = 0; i < lines.length; i++) { - const line = lines[i]; - if (line.indexOf("```") === 0) { - is_code = !is_code; - } - if (is_code) { - continue; - } - if (["- ", "- [ ] "].indexOf(line) > -1) - continue; - if (!line.startsWith("#") || ["#", " "].indexOf(line[1]) < 0) { - continue; - } - const heading_text = line.replace(/#/g, "").trim(); - const heading_index = block_headings.indexOf(heading_text); - if (heading_index < 0) - continue; - if (currentHeaders.length !== heading_index) - continue; - currentHeaders.push(heading_text); - if (currentHeaders.length === block_headings.length) { - if (heading_occurrence === 0) { - begin_line = i + 1; - break; - } - if (occurrence_count === heading_occurrence) { - begin_line = i + 1; - break; - } - occurrence_count++; - currentHeaders.pop(); - continue; - } - } - if (begin_line === 0) - return false; - is_code = false; - let char_count = 0; - for (i = begin_line; i < lines.length; i++) { - if (typeof line_limit === "number" && block.length > line_limit) { - block.push("..."); - break; - } - let line = lines[i]; - if (line.indexOf("#") === 0 && ["#", " "].indexOf(line[1]) !== -1) { - break; - } - if (limits.max_chars && char_count > limits.max_chars) { - block.push("..."); - break; - } - if (limits.max_chars && line.length + char_count > limits.max_chars) { - const max_new_chars = limits.max_chars - char_count; - line = line.slice(0, max_new_chars) + "..."; - break; - } - if (line.length === 0) - continue; - if (limits.chars_per_line && line.length > limits.chars_per_line) { - line = line.slice(0, limits.chars_per_line) + "..."; - } - if (line.startsWith("```")) { - is_code = !is_code; - continue; - } - if (is_code) { - line = " " + line; - } - block.push(line); - char_count += line.length; - } - if (is_code) { - block.push("```"); - } - return block.join("\n").trim(); - } - // retrieve a file from the vault - async file_retriever(link, limits = {}) { - limits = { - lines: null, - max_chars: null, - chars_per_line: null, - ...limits - }; - const this_file = this.app.vault.getAbstractFileByPath(link); - if (!(this_file instanceof Obsidian.TAbstractFile)) - return false; - const file_content = await this.app.vault.cachedRead(this_file); - const file_lines = file_content.split("\n"); - let first_ten_lines = []; - let is_code = false; - let char_accum = 0; - const line_limit2 = limits.lines || file_lines.length; - for (let i = 0; first_ten_lines.length < line_limit2; i++) { - let line = file_lines[i]; - if (typeof line === "undefined") - break; - if (line.length === 0) - continue; - if (limits.chars_per_line && line.length > limits.chars_per_line) { - line = line.slice(0, limits.chars_per_line) + "..."; - } - if (line === "---") - continue; - if (["- ", "- [ ] "].indexOf(line) > -1) - continue; - if (line.indexOf("```") === 0) { - is_code = !is_code; - continue; - } - if (limits.max_chars && char_accum > limits.max_chars) { - first_ten_lines.push("..."); - break; - } - if (is_code) { - line = " " + line; - } - if (line_is_heading(line)) { - if (first_ten_lines.length > 0 && line_is_heading(first_ten_lines[first_ten_lines.length - 1])) { - first_ten_lines.pop(); - } - } - first_ten_lines.push(line); - char_accum += line.length; - } - for (let i = 0; i < first_ten_lines.length; i++) { - if (line_is_heading(first_ten_lines[i])) { - if (i === first_ten_lines.length - 1) { - first_ten_lines.pop(); - break; - } - first_ten_lines[i] = first_ten_lines[i].replace(/#+/, ""); - first_ten_lines[i] = ` -${first_ten_lines[i]}:`; - } - } - first_ten_lines = first_ten_lines.join("\n"); - return first_ten_lines; - } - // iterate through blocks and skip if block_headings contains this.header_exclusions - validate_headings(block_headings) { - let valid = true; - if (this.header_exclusions.length > 0) { - for (let k = 0; k < this.header_exclusions.length; k++) { - if (block_headings.indexOf(this.header_exclusions[k]) > -1) { - valid = false; - this.log_exclusion("heading: " + this.header_exclusions[k]); - break; - } - } - } - return valid; - } - // render "Smart Connections" text fixed in the bottom right corner - render_brand(container, location = "default") { - if (container === "all") { - const locations = Object.keys(this.sc_branding); - for (let i = 0; i < locations.length; i++) { - this.render_brand(this.sc_branding[locations[i]], locations[i]); - } - return; - } - this.sc_branding[location] = container; - if (this.sc_branding[location].querySelector(".sc-brand")) { - this.sc_branding[location].querySelector(".sc-brand").remove(); - } - const brand_container = this.sc_branding[location].createEl("div", { cls: "sc-brand" }); - Obsidian.setIcon(brand_container, "smart-connections"); - const brand_p = brand_container.createEl("p"); - let text = "Smart Connections"; - let attr = {}; - if (this.update_available) { - text = "Update Available"; - attr = { - style: "font-weight: 700;" - }; - } - brand_p.createEl("a", { - cls: "", - text, - href: "https://github.com/brianpetro/obsidian-smart-connections/discussions", - target: "_blank", - attr - }); - } - // create list of nearest notes - async update_results(container, nearest) { - let list; - if (container.children.length > 1 && container.children[1].classList.contains("sc-list")) { - list = container.children[1]; - } - if (list) { - list.empty(); - } else { - list = container.createEl("div", { cls: "sc-list" }); - } - let search_result_class = "search-result"; - if (!this.settings.expanded_view) - search_result_class += " sc-collapsed"; - if (!this.settings.group_nearest_by_file) { - for (let i = 0; i < nearest.length; i++) { - console.log(this); - if (typeof nearest[i].link === "object") { - const item2 = list.createEl("div", { cls: "search-result" }); - const link2 = item2.createEl("a", { - cls: "search-result-file-title is-clickable", - href: nearest[i].link.path, - title: nearest[i].link.title - }); - link2.innerHTML = this.render_external_link_elm(nearest[i].link); - item2.setAttr("draggable", "true"); - continue; - } - let file_link_text; - const file_similarity_pct = Math.round(nearest[i].similarity * 100) + "%"; - if (this.settings.show_full_path) { - const pcs = nearest[i].link.split("/"); - file_link_text = pcs[pcs.length - 1]; - const path = pcs.slice(0, pcs.length - 1).join("/"); - file_link_text = `${file_similarity_pct} | ${path} | ${file_link_text}`; - } else { - file_link_text = "" + file_similarity_pct + " | " + nearest[i].link.split("/").pop() + ""; - } - if (!this.renderable_file_type(nearest[i].link)) { - const item2 = list.createEl("div", { cls: "search-result" }); - const link2 = item2.createEl("a", { - cls: "search-result-file-title is-clickable", - href: nearest[i].link - }); - link2.innerHTML = file_link_text; - item2.setAttr("draggable", "true"); - this.add_link_listeners(link2, nearest[i], item2); - continue; - } - file_link_text = file_link_text.replace(".md", "").replace(/#/g, " > "); - const item = list.createEl("div", { cls: search_result_class }); - const toggle = item.createEl("span", { cls: "is-clickable" }); - Obsidian.setIcon(toggle, "right-triangle"); - const link = toggle.createEl("a", { - cls: "search-result-file-title", - title: nearest[i].link - }); - link.innerHTML = file_link_text; - this.add_link_listeners(link, nearest[i], item); - toggle.addEventListener("click", (event) => { - let parent = event.target.parentElement; - while (!parent.classList.contains("search-result")) { - parent = parent.parentElement; - } - parent.classList.toggle("sc-collapsed"); - }); - const contents = item.createEl("ul", { cls: "" }); - const contents_container = contents.createEl("li", { - cls: "search-result-file-title is-clickable", - title: nearest[i].link - }); - if (nearest[i].link.indexOf("#") > -1) { - Obsidian.MarkdownRenderer.renderMarkdown(await this.block_retriever(nearest[i].link, { lines: 10, max_chars: 1e3 }), contents_container, nearest[i].link, new Obsidian.Component()); - } else { - const first_ten_lines = await this.file_retriever(nearest[i].link, { lines: 10, max_chars: 1e3 }); - if (!first_ten_lines) - continue; - Obsidian.MarkdownRenderer.renderMarkdown(first_ten_lines, contents_container, nearest[i].link, new Obsidian.Component()); - } - this.add_link_listeners(contents, nearest[i], item); - } - this.render_brand(container, "block"); - return; - } - const nearest_by_file = {}; - for (let i = 0; i < nearest.length; i++) { - const curr = nearest[i]; - const link = curr.link; - if (typeof link === "object") { - nearest_by_file[link.path] = [curr]; - continue; - } - if (link.indexOf("#") > -1) { - const file_path = link.split("#")[0]; - if (!nearest_by_file[file_path]) { - nearest_by_file[file_path] = []; - } - nearest_by_file[file_path].push(nearest[i]); - } else { - if (!nearest_by_file[link]) { - nearest_by_file[link] = []; - } - nearest_by_file[link].unshift(nearest[i]); - } - } - const keys = Object.keys(nearest_by_file); - for (let i = 0; i < keys.length; i++) { - const file = nearest_by_file[keys[i]]; - if (typeof file[0].link === "object") { - const curr = file[0]; - const meta = curr.link; - if (meta.path.startsWith("http")) { - const item2 = list.createEl("div", { cls: "search-result" }); - const link = item2.createEl("a", { - cls: "search-result-file-title is-clickable", - href: meta.path, - title: meta.title - }); - link.innerHTML = this.render_external_link_elm(meta); - item2.setAttr("draggable", "true"); - continue; - } - } - let file_link_text; - const file_similarity_pct = Math.round(file[0].similarity * 100) + "%"; - if (this.settings.show_full_path) { - const pcs = file[0].link.split("/"); - file_link_text = pcs[pcs.length - 1]; - const path = pcs.slice(0, pcs.length - 1).join("/"); - file_link_text = `${path} | ${file_similarity_pct}
${file_link_text}`; - } else { - file_link_text = file[0].link.split("/").pop(); - file_link_text += " | " + file_similarity_pct; - } - if (!this.renderable_file_type(file[0].link)) { - const item2 = list.createEl("div", { cls: "search-result" }); - const file_link2 = item2.createEl("a", { - cls: "search-result-file-title is-clickable", - title: file[0].link - }); - file_link2.innerHTML = file_link_text; - this.add_link_listeners(file_link2, file[0], item2); - continue; - } - file_link_text = file_link_text.replace(".md", "").replace(/#/g, " > "); - const item = list.createEl("div", { cls: search_result_class }); - const toggle = item.createEl("span", { cls: "is-clickable" }); - Obsidian.setIcon(toggle, "right-triangle"); - const file_link = toggle.createEl("a", { - cls: "search-result-file-title", - title: file[0].link - }); - file_link.innerHTML = file_link_text; - this.add_link_listeners(file_link, file[0], toggle); - toggle.addEventListener("click", (event) => { - let parent = event.target; - while (!parent.classList.contains("search-result")) { - parent = parent.parentElement; - } - parent.classList.toggle("sc-collapsed"); - }); - const file_link_list = item.createEl("ul"); - for (let j = 0; j < file.length; j++) { - if (file[j].link.indexOf("#") > -1) { - const block = file[j]; - const block_link = file_link_list.createEl("li", { - cls: "search-result-file-title is-clickable", - title: block.link - }); - if (file.length > 1) { - const block_context = this.render_block_context(block); - const block_similarity_pct = Math.round(block.similarity * 100) + "%"; - block_link.innerHTML = `${block_context} | ${block_similarity_pct}`; - } - const block_container = block_link.createEl("div"); - Obsidian.MarkdownRenderer.renderMarkdown(await this.block_retriever(block.link, { lines: 10, max_chars: 1e3 }), block_container, block.link, new Obsidian.Component()); - this.add_link_listeners(block_link, block, file_link_list); - } else { - const file_link_list2 = item.createEl("ul"); - const block_link = file_link_list2.createEl("li", { - cls: "search-result-file-title is-clickable", - title: file[0].link - }); - const block_container = block_link.createEl("div"); - let first_ten_lines = await this.file_retriever(file[0].link, { lines: 10, max_chars: 1e3 }); - if (!first_ten_lines) - continue; - Obsidian.MarkdownRenderer.renderMarkdown(first_ten_lines, block_container, file[0].link, new Obsidian.Component()); - this.add_link_listeners(block_link, file[0], file_link_list2); - } - } - } - this.render_brand(container, "file"); - } - add_link_listeners(item, curr, list) { - item.addEventListener("click", async (event) => { - await this.open_note(curr, event); - }); - item.setAttr("draggable", "true"); - item.addEventListener("dragstart", (event) => { - const dragManager = this.app.dragManager; - const file_path = curr.link.split("#")[0]; - const file = this.app.metadataCache.getFirstLinkpathDest(file_path, ""); - const dragData = dragManager.dragFile(event, file); - dragManager.onDragStart(event, dragData); - }); - if (curr.link.indexOf("{") > -1) - return; - item.addEventListener("mouseover", (event) => { - this.app.workspace.trigger("hover-link", { - event, - source: SMART_CONNECTIONS_VIEW_TYPE, - hoverParent: list, - targetEl: item, - linktext: curr.link - }); - }); - } - // get target file from link path - // if sub-section is linked, open file and scroll to sub-section - async open_note(curr, event = null) { - let targetFile; - let heading; - if (curr.link.indexOf("#") > -1) { - targetFile = this.app.metadataCache.getFirstLinkpathDest(curr.link.split("#")[0], ""); - const target_file_cache = this.app.metadataCache.getFileCache(targetFile); - let heading_text = curr.link.split("#").pop(); - let occurence = 0; - if (heading_text.indexOf("{") > -1) { - occurence = parseInt(heading_text.split("{")[1].split("}")[0]); - heading_text = heading_text.split("{")[0]; - } - const headings = target_file_cache.headings; - for (let i = 0; i < headings.length; i++) { - if (headings[i].heading === heading_text) { - if (occurence === 0) { - heading = headings[i]; - break; - } - occurence--; - } - } - } else { - targetFile = this.app.metadataCache.getFirstLinkpathDest(curr.link, ""); - } - let leaf; - if (event) { - const mod = Obsidian.Keymap.isModEvent(event); - leaf = this.app.workspace.getLeaf(mod); - } else { - leaf = this.app.workspace.getMostRecentLeaf(); - } - await leaf.openFile(targetFile); - if (heading) { - let { editor } = leaf.view; - const pos = { line: heading.position.start.line, ch: 0 }; - editor.setCursor(pos); - editor.scrollIntoView({ to: pos, from: pos }, true); - } - } - render_block_context(block) { - const block_headings = block.link.split(".md")[1].split("#"); - let block_context = ""; - for (let i = block_headings.length - 1; i >= 0; i--) { - if (block_context.length > 0) { - block_context = ` > ${block_context}`; - } - block_context = block_headings[i] + block_context; - if (block_context.length > 100) { - break; - } - } - if (block_context.startsWith(" > ")) { - block_context = block_context.slice(3); - } - return block_context; - } - renderable_file_type(link) { - return link.indexOf(".md") !== -1 && link.indexOf(".excalidraw") === -1; - } - render_external_link_elm(meta) { - if (meta.source) { - if (meta.source === "Gmail") - meta.source = "\u{1F4E7} Gmail"; - return `${meta.source}
${meta.title}`; - } - let domain = meta.path.replace(/(^\w+:|^)\/\//, ""); - domain = domain.split("/")[0]; - return `\u{1F310} ${domain}
${meta.title}`; - } - // get all folders - async get_all_folders() { - if (!this.folders || this.folders.length === 0) { - this.folders = await this.get_folders(); - } - return this.folders; - } - // get folders, traverse non-hidden sub-folders - async get_folders(path = "/") { - let folders = (await this.app.vault.adapter.list(path)).folders; - let folder_list = []; - for (let i = 0; i < folders.length; i++) { - if (folders[i].startsWith(".")) - continue; - folder_list.push(folders[i]); - folder_list = folder_list.concat(await this.get_folders(folders[i] + "/")); - } - return folder_list; - } - async sync_notes() { - if (!this.settings.license_key) { - new Obsidian.Notice("Smart Connections: Supporter license key is required to sync notes to the ChatGPT Plugin server."); - return; - } - console.log("syncing notes"); - const files = this.app.vault.getMarkdownFiles().filter((file) => { - for (let i = 0; i < this.file_exclusions.length; i++) { - if (file.path.indexOf(this.file_exclusions[i]) > -1) { - return false; - } - } - return true; - }); - const notes = await this.build_notes_object(files); - console.log("object built"); - await this.app.vault.adapter.write(".smart-connections/notes.json", JSON.stringify(notes, null, 2)); - console.log("notes saved"); - console.log(this.settings.license_key); - const response = await (0, Obsidian.requestUrl)({ - url: "https://sync.smartconnections.app/sync", - method: "POST", - headers: { - "Content-Type": "application/json" - }, - contentType: "application/json", - body: JSON.stringify({ - license_key: this.settings.license_key, - notes - }) - }); - console.log(response); - } - async build_notes_object(files) { - let output = {}; - for (let i = 0; i < files.length; i++) { - let file = files[i]; - let parts = file.path.split("/"); - let current = output; - for (let ii = 0; ii < parts.length; ii++) { - let part = parts[ii]; - if (ii === parts.length - 1) { - current[part] = await this.app.vault.cachedRead(file); - } else { - if (!current[part]) { - current[part] = {}; - } - current = current[part]; - } - } - } - return output; - } -}; -var SMART_CONNECTIONS_VIEW_TYPE = "smart-connections-view"; -var SmartConnectionsView = class extends Obsidian.ItemView { - constructor(leaf, plugin) { - super(leaf); - this.plugin = plugin; - this.nearest = null; - this.load_wait = null; - } - getViewType() { - return SMART_CONNECTIONS_VIEW_TYPE; - } - getDisplayText() { - return "Smart Connections Files"; - } - getIcon() { - return "smart-connections"; - } - set_message(message) { - const container = this.containerEl.children[1]; - container.empty(); - this.initiate_top_bar(container); - if (Array.isArray(message)) { - for (let i = 0; i < message.length; i++) { - container.createEl("p", { cls: "sc_message", text: message[i] }); - } - } else { - container.createEl("p", { cls: "sc_message", text: message }); - } - } - render_link_text(link, show_full_path = false) { - if (!show_full_path) { - link = link.split("/").pop(); - } - if (link.indexOf("#") > -1) { - link = link.split(".md"); - link[0] = `${link[0]}
`; - link = link.join(""); - link = link.replace(/\#/g, " \xBB "); - } else { - link = link.replace(".md", ""); - } - return link; - } - set_nearest(nearest, nearest_context = null, results_only = false) { - const container = this.containerEl.children[1]; - if (!results_only) { - container.empty(); - this.initiate_top_bar(container, nearest_context); - } - this.plugin.update_results(container, nearest); - } - initiate_top_bar(container, nearest_context = null) { - let top_bar; - if (container.children.length > 0 && container.children[0].classList.contains("sc-top-bar")) { - top_bar = container.children[0]; - top_bar.empty(); - } else { - top_bar = container.createEl("div", { cls: "sc-top-bar" }); - } - if (nearest_context) { - top_bar.createEl("p", { cls: "sc-context", text: nearest_context }); - } - const chat_button = top_bar.createEl("button", { cls: "sc-chat-button" }); - Obsidian.setIcon(chat_button, "message-square"); - chat_button.addEventListener("click", () => { - this.plugin.open_chat(); - }); - const search_button = top_bar.createEl("button", { cls: "sc-search-button" }); - Obsidian.setIcon(search_button, "search"); - search_button.addEventListener("click", () => { - top_bar.empty(); - const search_container = top_bar.createEl("div", { cls: "search-input-container" }); - const input = search_container.createEl("input", { - cls: "sc-search-input", - type: "search", - placeholder: "Type to start search..." - }); - input.focus(); - input.addEventListener("keydown", (event) => { - if (event.key === "Escape") { - this.clear_auto_searcher(); - this.initiate_top_bar(container, nearest_context); - } - }); - input.addEventListener("keyup", (event) => { - this.clear_auto_searcher(); - const search_term = input.value; - if (event.key === "Enter" && search_term !== "") { - this.search(search_term); - } else if (search_term !== "") { - clearTimeout(this.search_timeout); - this.search_timeout = setTimeout(() => { - this.search(search_term, true); - }, 700); - } - }); - }); - } - // render buttons: "create" and "retry" for loading embeddings.json file - render_embeddings_buttons() { - const container = this.containerEl.children[1]; - container.empty(); - container.createEl("h2", { cls: "scHeading", text: "Embeddings file not found" }); - const button_div = container.createEl("div", { cls: "scButtonDiv" }); - const create_button = button_div.createEl("button", { cls: "scButton", text: "Create embeddings.json" }); - button_div.createEl("p", { cls: "scButtonNote", text: "Warning: Creating embeddings.json file will trigger bulk embedding and may take a while" }); - const retry_button = button_div.createEl("button", { cls: "scButton", text: "Retry" }); - button_div.createEl("p", { cls: "scButtonNote", text: "If embeddings.json file already exists, click 'Retry' to load it" }); - create_button.addEventListener("click", async (event) => { - await this.plugin.smart_vec_lite.init_embeddings_file(); - await this.render_connections(); - }); - retry_button.addEventListener("click", async (event) => { - console.log("retrying to load embeddings.json file"); - await this.plugin.init_vecs(); - await this.render_connections(); - }); - } - async onOpen() { - const container = this.containerEl.children[1]; - container.empty(); - container.createEl("p", { cls: "scPlaceholder", text: "Open a note to find connections." }); - this.plugin.registerEvent(this.app.workspace.on("file-open", (file) => { - if (!file) { - return; - } - if (SUPPORTED_FILE_TYPES.indexOf(file.extension) === -1) { - return this.set_message([ - "File: " + file.name, - "Unsupported file type (Supported: " + SUPPORTED_FILE_TYPES.join(", ") + ")" - ]); - } - if (this.load_wait) { - clearTimeout(this.load_wait); - } - this.load_wait = setTimeout(() => { - this.render_connections(file); - this.load_wait = null; - }, 1e3); - })); - this.app.workspace.registerHoverLinkSource(SMART_CONNECTIONS_VIEW_TYPE, { - display: "Smart Connections Files", - defaultMod: true - }); - this.app.workspace.registerHoverLinkSource(SMART_CONNECTIONS_CHAT_VIEW_TYPE, { - display: "Smart Chat Links", - defaultMod: true - }); - this.app.workspace.onLayoutReady(this.initialize.bind(this)); - } - async initialize() { - this.set_message("Loading embeddings file..."); - const vecs_intiated = await this.plugin.init_vecs(); - if (vecs_intiated) { - this.set_message("Embeddings file loaded."); - await this.render_connections(); - } else { - this.render_embeddings_buttons(); - } - this.api = new SmartConnectionsViewApi(this.app, this.plugin, this); - (window["SmartConnectionsViewApi"] = this.api) && this.register(() => delete window["SmartConnectionsViewApi"]); - } - async onClose() { - console.log("closing smart connections view"); - this.app.workspace.unregisterHoverLinkSource(SMART_CONNECTIONS_VIEW_TYPE); - this.plugin.view = null; - } - async render_connections(context = null) { - console.log("rendering connections"); - if (!this.plugin.settings.api_key) { - this.set_message("An OpenAI API key is required to make Smart Connections"); - return; - } - if (!this.plugin.embeddings_loaded) { - await this.plugin.init_vecs(); - } - if (!this.plugin.embeddings_loaded) { - console.log("embeddings files still not loaded or yet to be created"); - this.render_embeddings_buttons(); - return; - } - this.set_message("Making Smart Connections..."); - if (typeof context === "string") { - const highlighted_text = context; - await this.search(highlighted_text); - return; - } - this.nearest = null; - this.interval_count = 0; - this.rendering = false; - this.file = context; - if (this.interval) { - clearInterval(this.interval); - this.interval = null; - } - this.interval = setInterval(() => { - if (!this.rendering) { - if (this.file instanceof Obsidian.TFile) { - this.rendering = true; - this.render_note_connections(this.file); - } else { - this.file = this.app.workspace.getActiveFile(); - if (!this.file && this.count > 1) { - clearInterval(this.interval); - this.set_message("No active file"); - return; - } - } - } else { - if (this.nearest) { - clearInterval(this.interval); - if (typeof this.nearest === "string") { - this.set_message(this.nearest); - } else { - this.set_nearest(this.nearest, "File: " + this.file.name); - } - if (this.plugin.render_log.failed_embeddings.length > 0) { - this.plugin.save_failed_embeddings(); - } - this.plugin.output_render_log(); - return; - } else { - this.interval_count++; - this.set_message("Making Smart Connections..." + this.interval_count); - } - } - }, 10); - } - async render_note_connections(file) { - this.nearest = await this.plugin.find_note_connections(file); - } - clear_auto_searcher() { - if (this.search_timeout) { - clearTimeout(this.search_timeout); - this.search_timeout = null; - } - } - async search(search_text, results_only = false) { - const nearest = await this.plugin.api.search(search_text); - const nearest_context = `Selection: "${search_text.length > 100 ? search_text.substring(0, 100) + "..." : search_text}"`; - this.set_nearest(nearest, nearest_context, results_only); - } -}; -var SmartConnectionsViewApi = class { - constructor(app, plugin, view) { - this.app = app; - this.plugin = plugin; - this.view = view; - } - async search(search_text) { - return await this.plugin.api.search(search_text); - } - // trigger reload of embeddings file - async reload_embeddings_file() { - await this.plugin.init_vecs(); - await this.view.render_connections(); - } - async init_vecs() { - this.smart_vec_lite = new VecLite({ - folder_path: ".smart-connections", - exists_adapter: this.app.vault.adapter.exists.bind( - this.app.vault.adapter - ), - mkdir_adapter: this.app.vault.adapter.mkdir.bind(this.app.vault.adapter), - read_adapter: this.app.vault.adapter.read.bind(this.app.vault.adapter), - rename_adapter: this.app.vault.adapter.rename.bind( - this.app.vault.adapter - ), - stat_adapter: this.app.vault.adapter.stat.bind(this.app.vault.adapter), - write_adapter: this.app.vault.adapter.write.bind(this.app.vault.adapter) - }); - this.embeddings_loaded = await this.smart_vec_lite.load(); - return this.embeddings_loaded; - } -}; -var ScSearchApi = class { - constructor(app, plugin) { - this.app = app; - this.plugin = plugin; - } - async search(search_text, filter = {}) { - filter = { - skip_sections: this.plugin.settings.skip_sections, - ...filter - }; - let nearest = []; - const resp = await this.plugin.request_embedding_from_input(search_text); - if (resp && resp.data && resp.data[0] && resp.data[0].embedding) { - nearest = this.plugin.smart_vec_lite.nearest(resp.data[0].embedding, filter); - } else { - new Obsidian.Notice("Smart Connections: Error getting embedding"); - } - return nearest; - } -}; -var SmartConnectionsSettingsTab = class extends Obsidian.PluginSettingTab { - constructor(app, plugin) { - super(app, plugin); - this.plugin = plugin; - } - display() { - const { - containerEl - } = this; - containerEl.empty(); - containerEl.createEl("h2", { - text: "Supporter Settings" - }); - containerEl.createEl("p", { - text: 'As a Smart Connections "Supporter", fast-track your PKM journey with priority perks and pioneering innovations.' - }); - const supporter_benefits_list = containerEl.createEl("ul"); - supporter_benefits_list.createEl("li", { - text: "Enjoy swift, top-priority support." - }); - supporter_benefits_list.createEl("li", { - text: "Gain early access to version 2 (includes local embedding model)." - }); - supporter_benefits_list.createEl("li", { - text: "Stay informed and engaged with exclusive supporter-only communications." - }); - new Obsidian.Setting(containerEl).setName("Supporter License Key").setDesc("Note: this is not required to use Smart Connections.").addText((text) => text.setPlaceholder("Enter your license_key").setValue(this.plugin.settings.license_key).onChange(async (value) => { - this.plugin.settings.license_key = value.trim(); - await this.plugin.saveSettings(true); - })); - new Obsidian.Setting(containerEl).setName("Get v2").setDesc("Get v2 (warning: very early beta release, likely to crash, please send issues directly to the supporter email for quick response)").addButton((button) => button.setButtonText("Get v2 (unstable)").onClick(async () => { - await this.plugin.update_to_v2(); - })); - new Obsidian.Setting(containerEl).setName("Sync Notes").setDesc("Make notes available via the Smart Connections ChatGPT Plugin. Respects exclusion settings configured below.").addButton((button) => button.setButtonText("Sync Notes").onClick(async () => { - await this.plugin.sync_notes(); - })); - new Obsidian.Setting(containerEl).setName("Become a Supporter").setDesc("Become a Supporter").addButton((button) => button.setButtonText("Become a Supporter").onClick(async () => { - const payment_pages = [ - "https://buy.stripe.com/9AQ5kO5QnbAWgGAbIY", - "https://buy.stripe.com/9AQ7sWemT48u1LGcN4" - ]; - if (!this.plugin.payment_page_index) { - this.plugin.payment_page_index = Math.round(Math.random()); - } - window.open(payment_pages[this.plugin.payment_page_index]); - })); - containerEl.createEl("h2", { - text: "OpenAI Settings" - }); - new Obsidian.Setting(containerEl).setName("OpenAI API Key").setDesc("Required: an OpenAI API key is currently required to use Smart Connections.").addText((text) => text.setPlaceholder("Enter your api_key").setValue(this.plugin.settings.api_key).onChange(async (value) => { - this.plugin.settings.api_key = value.trim(); - await this.plugin.saveSettings(true); - })); - new Obsidian.Setting(containerEl).setName("Test API Key").setDesc("Test API Key").addButton((button) => button.setButtonText("Test API Key").onClick(async () => { - const resp = await this.plugin.test_api_key(); - if (resp) { - new Obsidian.Notice("Smart Connections: API key is valid"); - } else { - new Obsidian.Notice("Smart Connections: API key is not working as expected!"); - } - })); - new Obsidian.Setting(containerEl).setName("Smart Chat Model").setDesc("Select a model to use with Smart Chat.").addDropdown((dropdown) => { - dropdown.addOption("gpt-3.5-turbo-16k", "gpt-3.5-turbo-16k"); - dropdown.addOption("gpt-4", "gpt-4 (limited access, 8k)"); - dropdown.addOption("gpt-3.5-turbo", "gpt-3.5-turbo (4k)"); - dropdown.addOption("gpt-4-1106-preview", "gpt-4-turbo (128k)"); - dropdown.onChange(async (value) => { - this.plugin.settings.smart_chat_model = value; - await this.plugin.saveSettings(); - }); - dropdown.setValue(this.plugin.settings.smart_chat_model); - }); - new Obsidian.Setting(containerEl).setName("Default Language").setDesc("Default language to use for Smart Chat. Changes which self-referential pronouns will trigger lookup of your notes.").addDropdown((dropdown) => { - const languages = Object.keys(SMART_TRANSLATION); - for (let i = 0; i < languages.length; i++) { - dropdown.addOption(languages[i], languages[i]); - } - dropdown.onChange(async (value) => { - this.plugin.settings.language = value; - await this.plugin.saveSettings(); - self_ref_pronouns_list.setText(this.get_self_ref_list()); - const chat_view = this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE).length > 0 ? this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE)[0].view : null; - if (chat_view) { - chat_view.new_chat(); - } - }); - dropdown.setValue(this.plugin.settings.language); - }); - const self_ref_pronouns_list = containerEl.createEl("span", { - text: this.get_self_ref_list() - }); - new Obsidian.Setting(containerEl).setName("Cut off frontmatter").setDesc("Cut off frontmatter in the prompt to gain characters in reply generation").addToggle((toggle) => { - toggle.setValue(this.plugin.settings.cut_off_frontmatter).onChange(async (value) => { - this.plugin.settings.cut_off_frontmatter = value; - await this.plugin.saveSettings(); - }); - }); - containerEl.createEl("h2", { - text: "Exclusions" - }); - new Obsidian.Setting(containerEl).setName("file_exclusions").setDesc("'Excluded file' matchers separated by a comma.").addText((text) => text.setPlaceholder("drawings,prompts/logs").setValue(this.plugin.settings.file_exclusions).onChange(async (value) => { - this.plugin.settings.file_exclusions = value; - await this.plugin.saveSettings(); - })); - new Obsidian.Setting(containerEl).setName("folder_exclusions").setDesc("'Excluded folder' matchers separated by a comma.").addText((text) => text.setPlaceholder("drawings,prompts/logs").setValue(this.plugin.settings.folder_exclusions).onChange(async (value) => { - this.plugin.settings.folder_exclusions = value; - await this.plugin.saveSettings(); - })); - new Obsidian.Setting(containerEl).setName("path_only").setDesc("'Path only' matchers separated by a comma.").addText((text) => text.setPlaceholder("drawings,prompts/logs").setValue(this.plugin.settings.path_only).onChange(async (value) => { - this.plugin.settings.path_only = value; - await this.plugin.saveSettings(); - })); - new Obsidian.Setting(containerEl).setName("header_exclusions").setDesc("'Excluded header' matchers separated by a comma. Works for 'blocks' only.").addText((text) => text.setPlaceholder("drawings,prompts/logs").setValue(this.plugin.settings.header_exclusions).onChange(async (value) => { - this.plugin.settings.header_exclusions = value; - await this.plugin.saveSettings(); - })); - containerEl.createEl("h2", { - text: "Display" - }); - new Obsidian.Setting(containerEl).setName("show_full_path").setDesc("Show full path in view.").addToggle((toggle) => toggle.setValue(this.plugin.settings.show_full_path).onChange(async (value) => { - this.plugin.settings.show_full_path = value; - await this.plugin.saveSettings(true); - })); - new Obsidian.Setting(containerEl).setName("expanded_view").setDesc("Expanded view by default.").addToggle((toggle) => toggle.setValue(this.plugin.settings.expanded_view).onChange(async (value) => { - this.plugin.settings.expanded_view = value; - await this.plugin.saveSettings(true); - })); - new Obsidian.Setting(containerEl).setName("group_nearest_by_file").setDesc("Group nearest by file.").addToggle((toggle) => toggle.setValue(this.plugin.settings.group_nearest_by_file).onChange(async (value) => { - this.plugin.settings.group_nearest_by_file = value; - await this.plugin.saveSettings(true); - })); - new Obsidian.Setting(containerEl).setName("view_open").setDesc("Open view on Obsidian startup.").addToggle((toggle) => toggle.setValue(this.plugin.settings.view_open).onChange(async (value) => { - this.plugin.settings.view_open = value; - await this.plugin.saveSettings(true); - })); - new Obsidian.Setting(containerEl).setName("chat_open").setDesc("Open view on Obsidian startup.").addToggle((toggle) => toggle.setValue(this.plugin.settings.chat_open).onChange(async (value) => { - this.plugin.settings.chat_open = value; - await this.plugin.saveSettings(true); - })); - new Obsidian.Setting(containerEl).setName("open_in_big_view").setDesc("Open in big view or small view.").addDropdown((dropdown) => { - dropdown.addOption(false, "Right pane (small)"); - dropdown.addOption(true, "Main pane (big)"); - dropdown.setValue(this.plugin.settings.open_in_big_view); - dropdown.onChange(async (value) => { - this.plugin.settings.open_in_big_view = JSON.parse(value); - await this.plugin.saveSettings(true); - this.plugin.open_chat(); - }); - }); - containerEl.createEl("h2", { - text: "Advanced" - }); - new Obsidian.Setting(containerEl).setName("log_render").setDesc("Log render details to console (includes token_usage).").addToggle((toggle) => toggle.setValue(this.plugin.settings.log_render).onChange(async (value) => { - this.plugin.settings.log_render = value; - await this.plugin.saveSettings(true); - })); - new Obsidian.Setting(containerEl).setName("log_render_files").setDesc("Log embedded objects paths with log render (for debugging).").addToggle((toggle) => toggle.setValue(this.plugin.settings.log_render_files).onChange(async (value) => { - this.plugin.settings.log_render_files = value; - await this.plugin.saveSettings(true); - })); - new Obsidian.Setting(containerEl).setName("skip_sections").setDesc("Skips making connections to specific sections within notes. Warning: reduces usefulness for large files and requires 'Force Refresh' for sections to work in the future.").addToggle((toggle) => toggle.setValue(this.plugin.settings.skip_sections).onChange(async (value) => { - this.plugin.settings.skip_sections = value; - await this.plugin.saveSettings(true); - })); - containerEl.createEl("h3", { - text: "Test File Writing" - }); - containerEl.createEl("h3", { - text: "Manual Save" - }); - let manual_save_results = containerEl.createEl("div"); - new Obsidian.Setting(containerEl).setName("manual_save").setDesc("Save current embeddings").addButton((button) => button.setButtonText("Manual Save").onClick(async () => { - if (confirm("Are you sure you want to save your current embeddings?")) { - try { - await this.plugin.save_embeddings_to_file(true); - manual_save_results.innerHTML = "Embeddings saved successfully."; - } catch (e) { - manual_save_results.innerHTML = "Embeddings failed to save. Error: " + e; - } - } - })); - containerEl.createEl("h3", { - text: "Previously failed files" - }); - let failed_list = containerEl.createEl("div"); - this.draw_failed_files_list(failed_list); - containerEl.createEl("h3", { - text: "Force Refresh" - }); - new Obsidian.Setting(containerEl).setName("force_refresh").setDesc("WARNING: DO NOT use unless you know what you are doing! This will delete all of your current embeddings from OpenAI and trigger reprocessing of your entire vault!").addButton((button) => button.setButtonText("Force Refresh").onClick(async () => { - if (confirm("Are you sure you want to Force Refresh? By clicking yes you confirm that you understand the consequences of this action.")) { - await this.plugin.force_refresh_embeddings_file(); - } - })); - } - get_self_ref_list() { - return "Current: " + SMART_TRANSLATION[this.plugin.settings.language].pronous.join(", "); - } - draw_failed_files_list(failed_list) { - failed_list.empty(); - if (this.plugin.settings.failed_files.length > 0) { - failed_list.createEl("p", { - text: "The following files failed to process and will be skipped until manually retried." - }); - let list = failed_list.createEl("ul"); - for (let failed_file of this.plugin.settings.failed_files) { - list.createEl("li", { - text: failed_file - }); - } - new Obsidian.Setting(failed_list).setName("retry_failed_files").setDesc("Retry failed files only").addButton((button) => button.setButtonText("Retry failed files only").onClick(async () => { - failed_list.empty(); - failed_list.createEl("p", { - text: "Retrying failed files..." - }); - await this.plugin.retry_failed_files(); - this.draw_failed_files_list(failed_list); - })); - } else { - failed_list.createEl("p", { - text: "No failed files" - }); - } - } -}; -function line_is_heading(line) { - return line.indexOf("#") === 0 && ["#", " "].indexOf(line[1]) !== -1; -} -var SMART_CONNECTIONS_CHAT_VIEW_TYPE = "smart-connections-chat-view"; -var SmartConnectionsChatView = class extends Obsidian.ItemView { - constructor(leaf, plugin) { - super(leaf); - this.plugin = plugin; - this.active_elm = null; - this.active_stream = null; - this.brackets_ct = 0; - this.chat = null; - this.chat_box = null; - this.chat_container = null; - this.current_chat_ml = []; - this.files = []; - this.last_from = null; - this.message_container = null; - this.prevent_input = false; - } - getDisplayText() { - return "Smart Connections Chat"; - } - getIcon() { - return "message-square"; - } - getViewType() { - return SMART_CONNECTIONS_CHAT_VIEW_TYPE; - } - onOpen() { - this.new_chat(); - this.plugin.get_all_folders(); - } - onClose() { - this.chat.save_chat(); - this.app.workspace.unregisterHoverLinkSource(SMART_CONNECTIONS_CHAT_VIEW_TYPE); - } - render_chat() { - this.containerEl.empty(); - this.chat_container = this.containerEl.createDiv("sc-chat-container"); - this.render_top_bar(); - this.render_chat_box(); - this.render_chat_input(); - this.plugin.render_brand(this.containerEl, "chat"); - } - // render plus sign for clear button - render_top_bar() { - let top_bar_container = this.chat_container.createDiv("sc-top-bar-container"); - let chat_name = this.chat.name(); - let chat_name_input = top_bar_container.createEl("input", { - attr: { - type: "text", - value: chat_name - }, - cls: "sc-chat-name-input" - }); - chat_name_input.addEventListener("change", this.rename_chat.bind(this)); - let smart_view_btn = this.create_top_bar_button(top_bar_container, "Smart View", "smart-connections"); - smart_view_btn.addEventListener("click", this.open_smart_view.bind(this)); - let save_btn = this.create_top_bar_button(top_bar_container, "Save Chat", "save"); - save_btn.addEventListener("click", this.save_chat.bind(this)); - let history_btn = this.create_top_bar_button(top_bar_container, "Chat History", "history"); - history_btn.addEventListener("click", this.open_chat_history.bind(this)); - const new_chat_btn = this.create_top_bar_button(top_bar_container, "New Chat", "plus"); - new_chat_btn.addEventListener("click", this.new_chat.bind(this)); - } - async open_chat_history() { - const folder = await this.app.vault.adapter.list(".smart-connections/chats"); - this.files = folder.files.map((file) => { - return file.replace(".smart-connections/chats/", "").replace(".json", ""); - }); - if (!this.modal) - this.modal = new SmartConnectionsChatHistoryModal(this.app, this); - this.modal.open(); - } - create_top_bar_button(top_bar_container, title, icon = null) { - let btn = top_bar_container.createEl("button", { - attr: { - title - } - }); - if (icon) { - Obsidian.setIcon(btn, icon); - } else { - btn.innerHTML = title; - } - return btn; - } - // render new chat - new_chat() { - this.clear_chat(); - this.render_chat(); - this.new_messsage_bubble("assistant"); - this.active_elm.innerHTML = "

" + SMART_TRANSLATION[this.plugin.settings.language].initial_message + "

"; - } - // open a chat from the chat history modal - async open_chat(chat_id) { - this.clear_chat(); - await this.chat.load_chat(chat_id); - this.render_chat(); - for (let i = 0; i < this.chat.chat_ml.length; i++) { - await this.render_message(this.chat.chat_ml[i].content, this.chat.chat_ml[i].role); - } - } - // clear current chat state - clear_chat() { - if (this.chat) { - this.chat.save_chat(); - } - this.chat = new SmartConnectionsChatModel(this.plugin); - if (this.dotdotdot_interval) { - clearInterval(this.dotdotdot_interval); - } - this.current_chat_ml = []; - this.end_stream(); - } - rename_chat(event) { - let new_chat_name = event.target.value; - this.chat.rename_chat(new_chat_name); - } - // save current chat - save_chat() { - this.chat.save_chat(); - new Obsidian.Notice("[Smart Connections] Chat saved"); - } - open_smart_view() { - this.plugin.open_view(); - } - // render chat messages container - render_chat_box() { - this.chat_box = this.chat_container.createDiv("sc-chat-box"); - this.message_container = this.chat_box.createDiv("sc-message-container"); - } - // open file suggestion modal - open_file_suggestion_modal() { - if (!this.file_selector) - this.file_selector = new SmartConnectionsFileSelectModal(this.app, this); - this.file_selector.open(); - } - // open folder suggestion modal - async open_folder_suggestion_modal() { - if (!this.folder_selector) { - this.folder_selector = new SmartConnectionsFolderSelectModal(this.app, this); - } - this.folder_selector.open(); - } - // insert_selection from file suggestion modal - insert_selection(insert_text) { - let caret_pos = this.textarea.selectionStart; - let text_before = this.textarea.value.substring(0, caret_pos); - let text_after = this.textarea.value.substring(caret_pos, this.textarea.value.length); - this.textarea.value = text_before + insert_text + text_after; - this.textarea.selectionStart = caret_pos + insert_text.length; - this.textarea.selectionEnd = caret_pos + insert_text.length; - this.textarea.focus(); - } - // render chat textarea and button - render_chat_input() { - let chat_input = this.chat_container.createDiv("sc-chat-form"); - this.textarea = chat_input.createEl("textarea", { - cls: "sc-chat-input", - attr: { - placeholder: SMART_TRANSLATION[this.plugin.settings.language].try_placeholder - } - }); - chat_input.addEventListener("keyup", (e) => { - if (["[", "/"].indexOf(e.key) === -1) - return; - const caret_pos = this.textarea.selectionStart; - if (e.key === "[") { - if (this.textarea.value[caret_pos - 2] === "[") { - this.open_file_suggestion_modal(); - return; - } - } else { - this.brackets_ct = 0; - } - if (e.key === "/") { - if (this.textarea.value.length === 1 || this.textarea.value[caret_pos - 2] === " ") { - this.open_folder_suggestion_modal(); - return; - } - } - }); - chat_input.addEventListener("keydown", (e) => { - if (e.key === "Enter" && e.shiftKey) { - e.preventDefault(); - if (this.prevent_input) { - console.log("wait until current response is finished"); - new Obsidian.Notice("[Smart Connections] Wait until current response is finished"); - return; - } - let user_input = this.textarea.value; - this.textarea.value = ""; - this.initialize_response(user_input); - } - this.textarea.style.height = "auto"; - this.textarea.style.height = this.textarea.scrollHeight + "px"; - }); - let button_container = chat_input.createDiv("sc-button-container"); - let abort_button = button_container.createEl("span", { attr: { id: "sc-abort-button", style: "display: none;" } }); - Obsidian.setIcon(abort_button, "square"); - abort_button.addEventListener("click", () => { - this.end_stream(); - }); - let button = button_container.createEl("button", { attr: { id: "sc-send-button" }, cls: "send-button" }); - button.innerHTML = "Send"; - button.addEventListener("click", () => { - if (this.prevent_input) { - console.log("wait until current response is finished"); - new Obsidian.Notice("Wait until current response is finished"); - return; - } - let user_input = this.textarea.value; - this.textarea.value = ""; - this.initialize_response(user_input); - }); - } - async initialize_response(user_input) { - this.set_streaming_ux(); - await this.render_message(user_input, "user"); - this.chat.new_message_in_thread({ - role: "user", - content: user_input - }); - await this.render_dotdotdot(); - if (this.chat.contains_internal_link(user_input)) { - this.chat.get_response_with_note_context(user_input, this); - return; - } - if (this.contains_self_referential_keywords(user_input) || this.chat.contains_folder_reference(user_input)) { - const context = await this.get_context_hyde(user_input); - const chatml = [ - { - role: "system", - // content: context_input - content: context - }, - { - role: "user", - content: user_input - } - ]; - this.request_chatgpt_completion({ messages: chatml, temperature: 0 }); - return; - } - this.request_chatgpt_completion(); - } - async render_dotdotdot() { - if (this.dotdotdot_interval) - clearInterval(this.dotdotdot_interval); - await this.render_message("...", "assistant"); - let dots = 0; - this.active_elm.innerHTML = "..."; - this.dotdotdot_interval = setInterval(() => { - dots++; - if (dots > 3) - dots = 1; - this.active_elm.innerHTML = ".".repeat(dots); - }, 500); - } - set_streaming_ux() { - this.prevent_input = true; - if (document.getElementById("sc-send-button")) - document.getElementById("sc-send-button").style.display = "none"; - if (document.getElementById("sc-abort-button")) - document.getElementById("sc-abort-button").style.display = "block"; - } - unset_streaming_ux() { - this.prevent_input = false; - if (document.getElementById("sc-send-button")) - document.getElementById("sc-send-button").style.display = ""; - if (document.getElementById("sc-abort-button")) - document.getElementById("sc-abort-button").style.display = "none"; - } - // check if includes keywords referring to one's own notes - contains_self_referential_keywords(user_input) { - const matches = user_input.match(this.plugin.self_ref_kw_regex); - if (matches) - return true; - return false; - } - // render message - async render_message(message, from = "assistant", append_last = false) { - if (this.dotdotdot_interval) { - clearInterval(this.dotdotdot_interval); - this.dotdotdot_interval = null; - this.active_elm.innerHTML = ""; - } - if (append_last) { - this.current_message_raw += message; - if (message.indexOf("\n") === -1) { - this.active_elm.innerHTML += message; - } else { - this.active_elm.innerHTML = ""; - await Obsidian.MarkdownRenderer.renderMarkdown(this.current_message_raw, this.active_elm, "?no-dataview", new Obsidian.Component()); - } - } else { - this.current_message_raw = ""; - if (this.chat.thread.length === 0 || this.last_from !== from) { - this.new_messsage_bubble(from); - } - this.active_elm.innerHTML = ""; - await Obsidian.MarkdownRenderer.renderMarkdown(message, this.active_elm, "?no-dataview", new Obsidian.Component()); - this.handle_links_in_message(); - this.render_message_action_buttons(message); - } - this.message_container.scrollTop = this.message_container.scrollHeight; - } - render_message_action_buttons(message) { - if (this.chat.context && this.chat.hyd) { - const context_view = this.active_elm.createEl("span", { - cls: "sc-msg-button", - attr: { - title: "Copy context to clipboard" - /* tooltip */ - } - }); - const this_hyd = this.chat.hyd; - Obsidian.setIcon(context_view, "eye"); - context_view.addEventListener("click", () => { - navigator.clipboard.writeText("```smart-connections\n" + this_hyd + "\n```\n"); - new Obsidian.Notice("[Smart Connections] Context code block copied to clipboard"); - }); - } - if (this.chat.context) { - const copy_prompt_button = this.active_elm.createEl("span", { - cls: "sc-msg-button", - attr: { - title: "Copy prompt to clipboard" - /* tooltip */ - } - }); - const this_context = this.chat.context.replace(/\`\`\`/g, " ```").trimLeft(); - Obsidian.setIcon(copy_prompt_button, "files"); - copy_prompt_button.addEventListener("click", () => { - navigator.clipboard.writeText("```prompt-context\n" + this_context + "\n```\n"); - new Obsidian.Notice("[Smart Connections] Context copied to clipboard"); - }); - } - const copy_button = this.active_elm.createEl("span", { - cls: "sc-msg-button", - attr: { - title: "Copy message to clipboard" - /* tooltip */ - } - }); - Obsidian.setIcon(copy_button, "copy"); - copy_button.addEventListener("click", () => { - navigator.clipboard.writeText(message.trimLeft()); - new Obsidian.Notice("[Smart Connections] Message copied to clipboard"); - }); - } - handle_links_in_message() { - const links = this.active_elm.querySelectorAll("a"); - if (links.length > 0) { - for (let i = 0; i < links.length; i++) { - const link = links[i]; - const link_text = link.getAttribute("data-href"); - link.addEventListener("mouseover", (event) => { - this.app.workspace.trigger("hover-link", { - event, - source: SMART_CONNECTIONS_CHAT_VIEW_TYPE, - hoverParent: link.parentElement, - targetEl: link, - // extract link text from a.data-href - linktext: link_text - }); - }); - link.addEventListener("click", (event) => { - const link_tfile = this.app.metadataCache.getFirstLinkpathDest(link_text, "/"); - const mod = Obsidian.Keymap.isModEvent(event); - let leaf = this.app.workspace.getLeaf(mod); - leaf.openFile(link_tfile); - }); - } - } - } - new_messsage_bubble(from) { - let message_el = this.message_container.createDiv(`sc-message ${from}`); - this.active_elm = message_el.createDiv("sc-message-content"); - this.last_from = from; - } - async request_chatgpt_completion(opts = {}) { - const chat_ml = opts.messages || opts.chat_ml || this.chat.prepare_chat_ml(); - console.log("chat_ml", chat_ml); - const max_total_tokens = Math.round(get_max_chars(this.plugin.settings.smart_chat_model) / 4); - console.log("max_total_tokens", max_total_tokens); - const curr_token_est = Math.round(JSON.stringify(chat_ml).length / 3); - console.log("curr_token_est", curr_token_est); - let max_available_tokens = max_total_tokens - curr_token_est; - if (max_available_tokens < 0) - max_available_tokens = 200; - else if (max_available_tokens > 4096) - max_available_tokens = 4096; - console.log("max_available_tokens", max_available_tokens); - opts = { - model: this.plugin.settings.smart_chat_model, - messages: chat_ml, - // max_tokens: 250, - max_tokens: max_available_tokens, - temperature: 0.3, - top_p: 1, - presence_penalty: 0, - frequency_penalty: 0, - stream: true, - stop: null, - n: 1, - // logit_bias: logit_bias, - ...opts - }; - if (opts.stream) { - const full_str = await new Promise((resolve, reject) => { - try { - const url = "https://api.openai.com/v1/chat/completions"; - this.active_stream = new ScStreamer(url, { - headers: { - "Content-Type": "application/json", - Authorization: `Bearer ${this.plugin.settings.api_key}` - }, - method: "POST", - payload: JSON.stringify(opts) - }); - let txt = ""; - this.active_stream.addEventListener("message", (e) => { - if (e.data != "[DONE]") { - const payload = JSON.parse(e.data); - const text = payload.choices[0].delta.content; - if (!text) { - return; - } - txt += text; - this.render_message(text, "assistant", true); - } else { - this.end_stream(); - resolve(txt); - } - }); - this.active_stream.addEventListener("readystatechange", (e) => { - if (e.readyState >= 2) { - console.log("ReadyState: " + e.readyState); - } - }); - this.active_stream.addEventListener("error", (e) => { - console.error(e); - new Obsidian.Notice("Smart Connections Error Streaming Response. See console for details."); - this.render_message("*API Error. See console logs for details.*", "assistant"); - this.end_stream(); - reject(e); - }); - this.active_stream.stream(); - } catch (err) { - console.error(err); - new Obsidian.Notice("Smart Connections Error Streaming Response. See console for details."); - this.end_stream(); - reject(err); - } - }); - await this.render_message(full_str, "assistant"); - this.chat.new_message_in_thread({ - role: "assistant", - content: full_str - }); - return; - } else { - try { - const response = await (0, Obsidian.requestUrl)({ - url: `https://api.openai.com/v1/chat/completions`, - method: "POST", - headers: { - Authorization: `Bearer ${this.plugin.settings.api_key}`, - "Content-Type": "application/json" - }, - contentType: "application/json", - body: JSON.stringify(opts), - throw: false - }); - return JSON.parse(response.text).choices[0].message.content; - } catch (err) { - new Obsidian.Notice(`Smart Connections API Error :: ${err}`); - } - } - } - end_stream() { - if (this.active_stream) { - this.active_stream.close(); - this.active_stream = null; - } - this.unset_streaming_ux(); - if (this.dotdotdot_interval) { - clearInterval(this.dotdotdot_interval); - this.dotdotdot_interval = null; - this.active_elm.parentElement.remove(); - this.active_elm = null; - } - } - async get_context_hyde(user_input) { - this.chat.reset_context(); - const hyd_input = `Anticipate what the user is seeking. Respond in the form of a hypothetical note written by the user. The note may contain statements as paragraphs, lists, or checklists in markdown format with no headings. Please respond with one hypothetical note and abstain from any other commentary. Use the format: PARENT FOLDER NAME > CHILD FOLDER NAME > FILE NAME > HEADING 1 > HEADING 2 > HEADING 3: HYPOTHETICAL NOTE CONTENTS.`; - const chatml = [ - { - role: "system", - content: hyd_input - }, - { - role: "user", - content: user_input - } - ]; - const hyd = await this.request_chatgpt_completion({ - messages: chatml, - stream: false, - temperature: 0, - max_tokens: 137 - }); - this.chat.hyd = hyd; - let filter = {}; - if (this.chat.contains_folder_reference(user_input)) { - const folder_refs = this.chat.get_folder_references(user_input); - if (folder_refs) { - filter = { - path_begins_with: folder_refs - }; - } - } - let nearest = await this.plugin.api.search(hyd, filter); - console.log("nearest", nearest.length); - nearest = this.get_nearest_until_next_dev_exceeds_std_dev(nearest); - console.log("nearest after std dev slice", nearest.length); - nearest = this.sort_by_len_adjusted_similarity(nearest); - return await this.get_context_for_prompt(nearest); - } - sort_by_len_adjusted_similarity(nearest) { - nearest = nearest.sort((a, b) => { - const a_score = a.similarity / a.len; - const b_score = b.similarity / b.len; - if (a_score > b_score) - return -1; - if (a_score < b_score) - return 1; - return 0; - }); - return nearest; - } - get_nearest_until_next_dev_exceeds_std_dev(nearest) { - const sim = nearest.map((n) => n.similarity); - const mean = sim.reduce((a, b) => a + b) / sim.length; - let std_dev = Math.sqrt(sim.map((x) => Math.pow(x - mean, 2)).reduce((a, b) => a + b) / sim.length); - let slice_i = 0; - while (slice_i < nearest.length) { - const next = nearest[slice_i + 1]; - if (next) { - const next_dev = Math.abs(next.similarity - nearest[slice_i].similarity); - if (next_dev > std_dev) { - if (slice_i < 3) - std_dev = std_dev * 1.5; - else - break; - } - } - slice_i++; - } - nearest = nearest.slice(0, slice_i + 1); - return nearest; - } - // this.test_get_nearest_until_next_dev_exceeds_std_dev(); - // // test get_nearest_until_next_dev_exceeds_std_dev - // test_get_nearest_until_next_dev_exceeds_std_dev() { - // const nearest = [{similarity: 0.99}, {similarity: 0.98}, {similarity: 0.97}, {similarity: 0.96}, {similarity: 0.95}, {similarity: 0.94}, {similarity: 0.93}, {similarity: 0.92}, {similarity: 0.91}, {similarity: 0.9}, {similarity: 0.79}, {similarity: 0.78}, {similarity: 0.77}, {similarity: 0.76}, {similarity: 0.75}, {similarity: 0.74}, {similarity: 0.73}, {similarity: 0.72}]; - // const result = this.get_nearest_until_next_dev_exceeds_std_dev(nearest); - // if(result.length !== 10){ - // console.error("get_nearest_until_next_dev_exceeds_std_dev failed", result); - // } - // } - async get_context_for_prompt(nearest) { - let context = []; - const MAX_SOURCES = this.plugin.settings.smart_chat_model === "gpt-4-1106-preview" ? 42 : 20; - const MAX_CHARS = get_max_chars(this.plugin.settings.smart_chat_model) / 2; - let char_accum = 0; - for (let i = 0; i < nearest.length; i++) { - if (context.length >= MAX_SOURCES) - break; - if (char_accum >= MAX_CHARS) - break; - if (typeof nearest[i].link !== "string") - continue; - const breadcrumbs = nearest[i].link.replace(/#/g, " > ").replace(".md", "").replace(/\//g, " > "); - let new_context = `${breadcrumbs}: -`; - const max_available_chars = MAX_CHARS - char_accum - new_context.length; - if (nearest[i].link.indexOf("#") !== -1) { - new_context += await this.plugin.block_retriever(nearest[i].link, { max_chars: max_available_chars }); - } else { - new_context += await this.plugin.file_retriever(nearest[i].link, { max_chars: max_available_chars }); - } - char_accum += new_context.length; - context.push({ - link: nearest[i].link, - text: new_context - }); - } - console.log("context sources: " + context.length); - console.log("total context tokens: ~" + Math.round(char_accum / 3.5)); - this.chat.context = `Anticipate the type of answer desired by the user. Imagine the following ${context.length} notes were written by the user and contain all the necessary information to answer the user's question. Begin responses with "${SMART_TRANSLATION[this.plugin.settings.language].prompt}..."`; - for (let i = 0; i < context.length; i++) { - this.chat.context += ` ----BEGIN #${i + 1}--- -${context[i].text} ----END #${i + 1}---`; - } - return this.chat.context; - } -}; -function get_max_chars(model = "gpt-3.5-turbo") { - const MAX_CHAR_MAP = { - "gpt-3.5-turbo-16k": 48e3, - "gpt-4": 24e3, - "gpt-3.5-turbo": 12e3, - "gpt-4-1106-preview": 2e5 - }; - return MAX_CHAR_MAP[model]; -} -var SmartConnectionsChatModel = class { - constructor(plugin) { - this.app = plugin.app; - this.plugin = plugin; - this.chat_id = null; - this.chat_ml = []; - this.context = null; - this.hyd = null; - this.thread = []; - } - async save_chat() { - if (this.thread.length === 0) - return; - if (!await this.app.vault.adapter.exists(".smart-connections/chats")) { - await this.app.vault.adapter.mkdir(".smart-connections/chats"); - } - if (!this.chat_id) { - this.chat_id = this.name() + "\u2014" + this.get_file_date_string(); - } - if (!this.chat_id.match(/^[a-zA-Z0-9_—\- ]+$/)) { - console.log("Invalid chat_id: " + this.chat_id); - new Obsidian.Notice("[Smart Connections] Failed to save chat. Invalid chat_id: '" + this.chat_id + "'"); - } - const chat_file = this.chat_id + ".json"; - this.app.vault.adapter.write( - ".smart-connections/chats/" + chat_file, - JSON.stringify(this.thread, null, 2) - ); - } - async load_chat(chat_id) { - this.chat_id = chat_id; - const chat_file = this.chat_id + ".json"; - let chat_json = await this.app.vault.adapter.read( - ".smart-connections/chats/" + chat_file - ); - this.thread = JSON.parse(chat_json); - this.chat_ml = this.prepare_chat_ml(); - } - // prepare chat_ml from chat - // gets the last message of each turn unless turn_variation_offsets=[[turn_index,variation_index]] is specified in offset - prepare_chat_ml(turn_variation_offsets = []) { - if (turn_variation_offsets.length === 0) { - this.chat_ml = this.thread.map((turn) => { - return turn[turn.length - 1]; - }); - } else { - let turn_variation_index = []; - for (let i = 0; i < turn_variation_offsets.length; i++) { - turn_variation_index[turn_variation_offsets[i][0]] = turn_variation_offsets[i][1]; - } - this.chat_ml = this.thread.map((turn, turn_index) => { - if (turn_variation_index[turn_index] !== void 0) { - return turn[turn_variation_index[turn_index]]; - } - return turn[turn.length - 1]; - }); - } - this.chat_ml = this.chat_ml.map((message) => { - return { - role: message.role, - content: message.content - }; - }); - return this.chat_ml; - } - last() { - return this.thread[this.thread.length - 1][this.thread[this.thread.length - 1].length - 1]; - } - last_from() { - return this.last().role; - } - // returns user_input or completion - last_message() { - return this.last().content; - } - // message={} - // add new message to thread - new_message_in_thread(message, turn = -1) { - if (this.context) { - message.context = this.context; - this.context = null; - } - if (this.hyd) { - message.hyd = this.hyd; - this.hyd = null; - } - if (turn === -1) { - this.thread.push([message]); - } else { - this.thread[turn].push(message); - } - } - reset_context() { - this.context = null; - this.hyd = null; - } - async rename_chat(new_name) { - if (this.chat_id && await this.app.vault.adapter.exists(".smart-connections/chats/" + this.chat_id + ".json")) { - new_name = this.chat_id.replace(this.name(), new_name); - await this.app.vault.adapter.rename( - ".smart-connections/chats/" + this.chat_id + ".json", - ".smart-connections/chats/" + new_name + ".json" - ); - this.chat_id = new_name; - } else { - this.chat_id = new_name + "\u2014" + this.get_file_date_string(); - await this.save_chat(); - } - } - name() { - if (this.chat_id) { - return this.chat_id.replace(/—[^—]*$/, ""); - } - return "UNTITLED"; - } - get_file_date_string() { - return (/* @__PURE__ */ new Date()).toISOString().replace(/(T|:|\..*)/g, " ").trim(); - } - // get response from with note context - async get_response_with_note_context(user_input, chat_view) { - let system_input = "Imagine the following notes were written by the user and contain the necessary information to synthesize a useful answer the user's query:\n"; - const notes = this.extract_internal_links(user_input); - let max_chars = get_max_chars(this.plugin.settings.smart_chat_model); - for (let i = 0; i < notes.length; i++) { - const this_max_chars = notes.length - i > 1 ? Math.floor(max_chars / (notes.length - i)) : max_chars; - const note_content = await this.get_note_contents(notes[i], { char_limit: this_max_chars }); - console.log(note_content); - system_input += `---BEGIN NOTE: [[${notes[i].basename}]]--- -`; - system_input += note_content; - system_input += `---END NOTE--- -`; - max_chars -= note_content.length; - if (max_chars <= 0) - break; - } - this.context = system_input; - const chatml = [ - { - role: "system", - content: system_input - }, - { - role: "user", - content: user_input - } - ]; - chat_view.request_chatgpt_completion({ messages: chatml, temperature: 0 }); - } - // check if contains internal link - contains_internal_link(user_input) { - if (user_input.indexOf("[[") === -1) - return false; - if (user_input.indexOf("]]") === -1) - return false; - return true; - } - // check if contains folder reference (ex. /folder/, or /folder/subfolder/) - contains_folder_reference(user_input) { - if (user_input.indexOf("/") === -1) - return false; - if (user_input.indexOf("/") === user_input.lastIndexOf("/")) - return false; - return true; - } - // get folder references from user input - get_folder_references(user_input) { - const folders = this.plugin.folders.slice(); - const matches = folders.sort((a, b) => b.length - a.length).map((folder) => { - if (user_input.indexOf(folder) !== -1) { - user_input = user_input.replace(folder, ""); - return folder; - } - return false; - }).filter((folder) => folder); - console.log(matches); - if (matches) - return matches; - return false; - } - // extract internal links - extract_internal_links(user_input) { - const matches = user_input.match(/\[\[(.*?)\]\]/g); - console.log(matches); - if (matches) - return matches.map((match) => { - return this.app.metadataCache.getFirstLinkpathDest(match.replace("[[", "").replace("]]", ""), "/"); - }); - return []; - } - // get context from internal links - async get_note_contents(note, opts = {}) { - opts = { - char_limit: 1e4, - ...opts - }; - if (!(note instanceof Obsidian.TFile)) - return ""; - let file_content = await this.app.vault.cachedRead(note); - if (this.plugin.settings.cut_off_frontmatter) { - file_content = file_content.replace(/\s*---[\s\S]*?---/, ""); - } - if (file_content.indexOf("```dataview") > -1) { - file_content = await this.render_dataview_queries(file_content, note.path, opts); - } - return file_content.substring(0, opts.char_limit); - } - async render_dataview_queries(file_content, note_path, opts = {}) { - opts = { - char_limit: null, - ...opts - }; - const dataview_api = window["DataviewAPI"]; - if (!dataview_api) - return file_content; - const dataview_code_blocks = file_content.match(/```dataview(.*?)```/gs); - for (let i = 0; i < dataview_code_blocks.length; i++) { - if (opts.char_limit && opts.char_limit < file_content.indexOf(dataview_code_blocks[i])) - break; - const dataview_code_block = dataview_code_blocks[i]; - const dataview_code_block_content = dataview_code_block.replace("```dataview", "").replace("```", ""); - const dataview_query_result = await dataview_api.queryMarkdown(dataview_code_block_content, note_path, null); - if (dataview_query_result.successful) { - file_content = file_content.replace(dataview_code_block, dataview_query_result.value); - } - } - return file_content; - } -}; -var SmartConnectionsChatHistoryModal = class extends Obsidian.FuzzySuggestModal { - constructor(app, view, files) { - super(app); - this.app = app; - this.view = view; - this.setPlaceholder("Type the name of a chat session..."); - } - getItems() { - if (!this.view.files) { - return []; - } - return this.view.files; - } - getItemText(item) { - if (item.indexOf("UNTITLED") === -1) { - item.replace(/—[^—]*$/, ""); - } - return item; - } - onChooseItem(session) { - this.view.open_chat(session); - } -}; -var SmartConnectionsFileSelectModal = class extends Obsidian.FuzzySuggestModal { - constructor(app, view) { - super(app); - this.app = app; - this.view = view; - this.setPlaceholder("Type the name of a file..."); - } - getItems() { - return this.app.vault.getMarkdownFiles().sort((a, b) => a.basename.localeCompare(b.basename)); - } - getItemText(item) { - return item.basename; - } - onChooseItem(file) { - this.view.insert_selection(file.basename + "]] "); - } -}; -var SmartConnectionsFolderSelectModal = class extends Obsidian.FuzzySuggestModal { - constructor(app, view) { - super(app); - this.app = app; - this.view = view; - this.setPlaceholder("Type the name of a folder..."); - } - getItems() { - return this.view.plugin.folders; - } - getItemText(item) { - return item; - } - onChooseItem(folder) { - this.view.insert_selection(folder + "/ "); - } -}; -var ScStreamer = class { - // constructor - constructor(url, options) { - options = options || {}; - this.url = url; - this.method = options.method || "GET"; - this.headers = options.headers || {}; - this.payload = options.payload || null; - this.withCredentials = options.withCredentials || false; - this.listeners = {}; - this.readyState = this.CONNECTING; - this.progress = 0; - this.chunk = ""; - this.xhr = null; - this.FIELD_SEPARATOR = ":"; - this.INITIALIZING = -1; - this.CONNECTING = 0; - this.OPEN = 1; - this.CLOSED = 2; - } - // addEventListener - addEventListener(type, listener) { - if (!this.listeners[type]) { - this.listeners[type] = []; - } - if (this.listeners[type].indexOf(listener) === -1) { - this.listeners[type].push(listener); - } - } - // removeEventListener - removeEventListener(type, listener) { - if (!this.listeners[type]) { - return; - } - let filtered = []; - for (let i = 0; i < this.listeners[type].length; i++) { - if (this.listeners[type][i] !== listener) { - filtered.push(this.listeners[type][i]); - } - } - if (this.listeners[type].length === 0) { - delete this.listeners[type]; - } else { - this.listeners[type] = filtered; - } - } - // dispatchEvent - dispatchEvent(event) { - if (!event) { - return true; - } - event.source = this; - let onHandler = "on" + event.type; - if (this.hasOwnProperty(onHandler)) { - this[onHandler].call(this, event); - if (event.defaultPrevented) { - return false; - } - } - if (this.listeners[event.type]) { - return this.listeners[event.type].every(function(callback) { - callback(event); - return !event.defaultPrevented; - }); - } - return true; - } - // _setReadyState - _setReadyState(state) { - let event = new CustomEvent("readyStateChange"); - event.readyState = state; - this.readyState = state; - this.dispatchEvent(event); - } - // _onStreamFailure - _onStreamFailure(e) { - let event = new CustomEvent("error"); - event.data = e.currentTarget.response; - this.dispatchEvent(event); - this.close(); - } - // _onStreamAbort - _onStreamAbort(e) { - let event = new CustomEvent("abort"); - this.close(); - } - // _onStreamProgress - _onStreamProgress(e) { - if (!this.xhr) { - return; - } - if (this.xhr.status !== 200) { - this._onStreamFailure(e); - return; - } - if (this.readyState === this.CONNECTING) { - this.dispatchEvent(new CustomEvent("open")); - this._setReadyState(this.OPEN); - } - let data = this.xhr.responseText.substring(this.progress); - this.progress += data.length; - data.split(/(\r\n|\r|\n){2}/g).forEach(function(part) { - if (part.trim().length === 0) { - this.dispatchEvent(this._parseEventChunk(this.chunk.trim())); - this.chunk = ""; - } else { - this.chunk += part; - } - }.bind(this)); - } - // _onStreamLoaded - _onStreamLoaded(e) { - this._onStreamProgress(e); - this.dispatchEvent(this._parseEventChunk(this.chunk)); - this.chunk = ""; - } - // _parseEventChunk - _parseEventChunk(chunk) { - if (!chunk || chunk.length === 0) { - return null; - } - let e = { id: null, retry: null, data: "", event: "message" }; - chunk.split(/(\r\n|\r|\n)/).forEach(function(line) { - line = line.trimRight(); - let index = line.indexOf(this.FIELD_SEPARATOR); - if (index <= 0) { - return; - } - let field = line.substring(0, index); - if (!(field in e)) { - return; - } - let value = line.substring(index + 1).trimLeft(); - if (field === "data") { - e[field] += value; - } else { - e[field] = value; - } - }.bind(this)); - let event = new CustomEvent(e.event); - event.data = e.data; - event.id = e.id; - return event; - } - // _checkStreamClosed - _checkStreamClosed() { - if (!this.xhr) { - return; - } - if (this.xhr.readyState === XMLHttpRequest.DONE) { - this._setReadyState(this.CLOSED); - } - } - // stream - stream() { - this._setReadyState(this.CONNECTING); - this.xhr = new XMLHttpRequest(); - this.xhr.addEventListener("progress", this._onStreamProgress.bind(this)); - this.xhr.addEventListener("load", this._onStreamLoaded.bind(this)); - this.xhr.addEventListener("readystatechange", this._checkStreamClosed.bind(this)); - this.xhr.addEventListener("error", this._onStreamFailure.bind(this)); - this.xhr.addEventListener("abort", this._onStreamAbort.bind(this)); - this.xhr.open(this.method, this.url); - for (let header in this.headers) { - this.xhr.setRequestHeader(header, this.headers[header]); - } - this.xhr.withCredentials = this.withCredentials; - this.xhr.send(this.payload); - } - // close - close() { - if (this.readyState === this.CLOSED) { - return; - } - this.xhr.abort(); - this.xhr = null; - this._setReadyState(this.CLOSED); - } -}; -module.exports = SmartConnectionsPlugin; -//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/index.js"],
  "sourcesContent": ["const Obsidian = require(\"obsidian\");\r\nconst VecLite = class {\r\n  constructor(config) {\r\n    this.config = {\r\n      file_name: \"embeddings-3.json\",\r\n      folder_path: \".vec_lite\",\r\n      exists_adapter: null,\r\n      mkdir_adapter: null,\r\n      read_adapter: null,\r\n      rename_adapter: null,\r\n      stat_adapter: null,\r\n      write_adapter: null,\r\n      ...config,\r\n    };\r\n    this.file_name = this.config.file_name;\r\n    this.folder_path = config.folder_path;\r\n    this.file_path = this.folder_path + \"/\" + this.file_name;\r\n    this.embeddings = false;\r\n  }\r\n  async file_exists(path) {\r\n    if (this.config.exists_adapter) {\r\n      return await this.config.exists_adapter(path);\r\n    } else {\r\n      throw new Error(\"exists_adapter not set\");\r\n    }\r\n  }\r\n  async mkdir(path) {\r\n    if (this.config.mkdir_adapter) {\r\n      return await this.config.mkdir_adapter(path);\r\n    } else {\r\n      throw new Error(\"mkdir_adapter not set\");\r\n    }\r\n  }\r\n  async read_file(path) {\r\n    if (this.config.read_adapter) {\r\n      return await this.config.read_adapter(path);\r\n    } else {\r\n      throw new Error(\"read_adapter not set\");\r\n    }\r\n  }\r\n  async rename(old_path, new_path) {\r\n    if (this.config.rename_adapter) {\r\n      return await this.config.rename_adapter(old_path, new_path);\r\n    } else {\r\n      throw new Error(\"rename_adapter not set\");\r\n    }\r\n  }\r\n  async stat(path) {\r\n    if (this.config.stat_adapter) {\r\n      return await this.config.stat_adapter(path);\r\n    } else {\r\n      throw new Error(\"stat_adapter not set\");\r\n    }\r\n  }\r\n  async write_file(path, data) {\r\n    if (this.config.write_adapter) {\r\n      return await this.config.write_adapter(path, data);\r\n    } else {\r\n      throw new Error(\"write_adapter not set\");\r\n    }\r\n  }\r\n  async load(retries = 0) {\r\n    try {\r\n      const embeddings_file = await this.read_file(this.file_path);\r\n      this.embeddings = JSON.parse(embeddings_file);\r\n      console.log(\"loaded embeddings file: \" + this.file_path);\r\n      return true;\r\n    } catch (error) {\r\n      if (retries < 3) {\r\n        console.log(\"retrying load()\");\r\n        await new Promise((r) => setTimeout(r, 1e3 + 1e3 * retries));\r\n        return await this.load(retries + 1);\r\n      }\r\n      console.log(\r\n        \"failed to load embeddings file, prompt user to initiate bulk embed\"\r\n      );\r\n      return false;\r\n    }\r\n  }\r\n  async init_embeddings_file() {\r\n    if (!(await this.file_exists(this.folder_path))) {\r\n      await this.mkdir(this.folder_path);\r\n      console.log(\"created folder: \" + this.folder_path);\r\n    } else {\r\n      console.log(\"folder already exists: \" + this.folder_path);\r\n    }\r\n    if (!(await this.file_exists(this.file_path))) {\r\n      await this.write_file(this.file_path, \"{}\");\r\n      console.log(\"created embeddings file: \" + this.file_path);\r\n    } else {\r\n      console.log(\"embeddings file already exists: \" + this.file_path);\r\n    }\r\n  }\r\n  async save() {\r\n    const embeddings = JSON.stringify(this.embeddings);\r\n    const embeddings_file_exists = await this.file_exists(this.file_path);\r\n    if (embeddings_file_exists) {\r\n      const new_file_size = embeddings.length;\r\n      const existing_file_size = await this.stat(this.file_path).then(\r\n        (stat) => stat.size\r\n      );\r\n      if (new_file_size > existing_file_size * 0.5) {\r\n        await this.write_file(this.file_path, embeddings);\r\n        console.log(\"embeddings file size: \" + new_file_size + \" bytes\");\r\n      } else {\r\n        const warning_message = [\r\n          \"Warning: New embeddings file size is significantly smaller than existing embeddings file size.\",\r\n          \"Aborting to prevent possible loss of embeddings data.\",\r\n          \"New file size: \" + new_file_size + \" bytes.\",\r\n          \"Existing file size: \" + existing_file_size + \" bytes.\",\r\n          \"Restarting Obsidian may fix this.\",\r\n        ];\r\n        console.log(warning_message.join(\" \"));\r\n        await this.write_file(\r\n          this.folder_path + \"/unsaved-embeddings.json\",\r\n          embeddings\r\n        );\r\n        throw new Error(\r\n          \"Error: New embeddings file size is significantly smaller than existing embeddings file size. Aborting to prevent possible loss of embeddings data.\"\r\n        );\r\n      }\r\n    } else {\r\n      await this.init_embeddings_file();\r\n      return await this.save();\r\n    }\r\n    return true;\r\n  }\r\n  cos_sim(vector1, vector2) {\r\n    let dotProduct = 0;\r\n    let normA = 0;\r\n    let normB = 0;\r\n    for (let i = 0; i < vector1.length; i++) {\r\n      dotProduct += vector1[i] * vector2[i];\r\n      normA += vector1[i] * vector1[i];\r\n      normB += vector2[i] * vector2[i];\r\n    }\r\n    if (normA === 0 || normB === 0) {\r\n      return 0;\r\n    } else {\r\n      return dotProduct / (Math.sqrt(normA) * Math.sqrt(normB));\r\n    }\r\n  }\r\n  nearest(to_vec, filter = {}) {\r\n    filter = {\r\n      results_count: 30,\r\n      ...filter,\r\n    };\r\n    let nearest = [];\r\n    const from_keys = Object.keys(this.embeddings);\r\n    for (let i = 0; i < from_keys.length; i++) {\r\n      if (filter.skip_sections) {\r\n        const from_path = this.embeddings[from_keys[i]].meta.path;\r\n        if (from_path.indexOf(\"#\") > -1) continue;\r\n      }\r\n      if (filter.skip_key) {\r\n        if (filter.skip_key === from_keys[i]) continue;\r\n        if (filter.skip_key === this.embeddings[from_keys[i]].meta.parent)\r\n          continue;\r\n      }\r\n      if (filter.path_begins_with) {\r\n        if (\r\n          typeof filter.path_begins_with === \"string\" &&\r\n          !this.embeddings[from_keys[i]].meta.path.startsWith(\r\n            filter.path_begins_with\r\n          )\r\n        )\r\n          continue;\r\n        if (\r\n          Array.isArray(filter.path_begins_with) &&\r\n          !filter.path_begins_with.some((path) =>\r\n            this.embeddings[from_keys[i]].meta.path.startsWith(path)\r\n          )\r\n        )\r\n          continue;\r\n      }\r\n      nearest.push({\r\n        link: this.embeddings[from_keys[i]].meta.path,\r\n        similarity: this.cos_sim(to_vec, this.embeddings[from_keys[i]].vec),\r\n        size: this.embeddings[from_keys[i]].meta.size,\r\n      });\r\n    }\r\n    nearest.sort(function (a, b) {\r\n      return b.similarity - a.similarity;\r\n    });\r\n    nearest = nearest.slice(0, filter.results_count);\r\n    return nearest;\r\n  }\r\n  find_nearest_embeddings(to_vec, filter = {}) {\r\n    const default_filter = {\r\n      max: this.max_sources,\r\n    };\r\n    filter = { ...default_filter, ...filter };\r\n    if (Array.isArray(to_vec) && to_vec.length !== this.vec_len) {\r\n      this.nearest = {};\r\n      for (let i = 0; i < to_vec.length; i++) {\r\n        this.find_nearest_embeddings(to_vec[i], {\r\n          max: Math.floor(filter.max / to_vec.length),\r\n        });\r\n      }\r\n    } else {\r\n      const from_keys = Object.keys(this.embeddings);\r\n      for (let i = 0; i < from_keys.length; i++) {\r\n        if (this.validate_type(this.embeddings[from_keys[i]])) continue;\r\n        const sim = this.computeCosineSimilarity(\r\n          to_vec,\r\n          this.embeddings[from_keys[i]].vec\r\n        );\r\n        if (this.nearest[from_keys[i]]) {\r\n          this.nearest[from_keys[i]] += sim;\r\n        } else {\r\n          this.nearest[from_keys[i]] = sim;\r\n        }\r\n      }\r\n    }\r\n    let nearest = Object.keys(this.nearest).map((key) => {\r\n      return {\r\n        key,\r\n        similarity: this.nearest[key],\r\n      };\r\n    });\r\n    nearest = this.sort_by_similarity(nearest);\r\n    nearest = nearest.slice(0, filter.max);\r\n    nearest = nearest.map((item) => {\r\n      return {\r\n        link: this.embeddings[item.key].meta.path,\r\n        similarity: item.similarity,\r\n        len:\r\n          this.embeddings[item.key].meta.len ||\r\n          this.embeddings[item.key].meta.size,\r\n      };\r\n    });\r\n    return nearest;\r\n  }\r\n  sort_by_similarity(nearest) {\r\n    return nearest.sort(function (a, b) {\r\n      const a_score = a.similarity;\r\n      const b_score = b.similarity;\r\n      if (a_score > b_score) return -1;\r\n      if (a_score < b_score) return 1;\r\n      return 0;\r\n    });\r\n  }\r\n  // check if key from embeddings exists in files\r\n  clean_up_embeddings(files) {\r\n    console.log(\"cleaning up embeddings\");\r\n    const keys = Object.keys(this.embeddings);\r\n    let deleted_embeddings = 0;\r\n    for (const key of keys) {\r\n      const path = this.embeddings[key].meta.path;\r\n      if (!files.find((file) => path.startsWith(file.path))) {\r\n        delete this.embeddings[key];\r\n        deleted_embeddings++;\r\n        continue;\r\n      }\r\n      if (path.indexOf(\"#\") > -1) {\r\n        const parent_key = this.embeddings[key].meta.parent;\r\n        if (!this.embeddings[parent_key]) {\r\n          delete this.embeddings[key];\r\n          deleted_embeddings++;\r\n          continue;\r\n        }\r\n        if (!this.embeddings[parent_key].meta) {\r\n          delete this.embeddings[key];\r\n          deleted_embeddings++;\r\n          continue;\r\n        }\r\n        if (\r\n          this.embeddings[parent_key].meta.children &&\r\n          this.embeddings[parent_key].meta.children.indexOf(key) < 0\r\n        ) {\r\n          delete this.embeddings[key];\r\n          deleted_embeddings++;\r\n          continue;\r\n        }\r\n      }\r\n    }\r\n    return { deleted_embeddings, total_embeddings: keys.length };\r\n  }\r\n  get(key) {\r\n    return this.embeddings[key] || null;\r\n  }\r\n  get_meta(key) {\r\n    const embedding = this.get(key);\r\n    if (embedding && embedding.meta) {\r\n      return embedding.meta;\r\n    }\r\n    return null;\r\n  }\r\n  get_mtime(key) {\r\n    const meta = this.get_meta(key);\r\n    if (meta && meta.mtime) {\r\n      return meta.mtime;\r\n    }\r\n    return null;\r\n  }\r\n  get_hash(key) {\r\n    const meta = this.get_meta(key);\r\n    if (meta && meta.hash) {\r\n      return meta.hash;\r\n    }\r\n    return null;\r\n  }\r\n  get_size(key) {\r\n    const meta = this.get_meta(key);\r\n    if (meta && meta.size) {\r\n      return meta.size;\r\n    }\r\n    return null;\r\n  }\r\n  get_children(key) {\r\n    const meta = this.get_meta(key);\r\n    if (meta && meta.children) {\r\n      return meta.children;\r\n    }\r\n    return null;\r\n  }\r\n  get_vec(key) {\r\n    const embedding = this.get(key);\r\n    if (embedding && embedding.vec) {\r\n      return embedding.vec;\r\n    }\r\n    return null;\r\n  }\r\n  save_embedding(key, vec, meta) {\r\n    this.embeddings[key] = {\r\n      vec,\r\n      meta,\r\n    };\r\n  }\r\n  mtime_is_current(key, source_mtime) {\r\n    const mtime = this.get_mtime(key);\r\n    if (mtime && mtime >= source_mtime) {\r\n      return true;\r\n    }\r\n    return false;\r\n  }\r\n  async force_refresh() {\r\n    this.embeddings = null;\r\n    this.embeddings = {};\r\n    let current_datetime = Math.floor(Date.now() / 1e3);\r\n    await this.rename(\r\n      this.file_path,\r\n      this.folder_path + \"/embeddings-\" + current_datetime + \".json\"\r\n    );\r\n    await this.init_embeddings_file();\r\n  }\r\n};\r\n\r\n\r\n\r\nconst DEFAULT_SETTINGS = {\r\n  api_key: \"\",\r\n  chat_open: true,\r\n  file_exclusions: \"\",\r\n  folder_exclusions: \"\",\r\n  header_exclusions: \"\",\r\n  path_only: \"\",\r\n  show_full_path: false,\r\n  cut_off_frontmatter: false,\r\n  expanded_view: true,\r\n  group_nearest_by_file: false,\r\n  language: \"en\",\r\n  log_render: false,\r\n  log_render_files: false,\r\n  recently_sent_retry_notice: false,\r\n  skip_sections: false,\r\n  smart_chat_model: \"gpt-3.5-turbo-16k\",\r\n  view_open: true,\r\n  version: \"\",\r\n  open_in_big_view: false,\r\n};\r\nconst MAX_EMBED_STRING_LENGTH = 25000;\r\n\r\nlet VERSION;\r\nconst SUPPORTED_FILE_TYPES = [\"md\", \"canvas\"];\r\n\r\n//create one object with all the translations\r\n// research : SMART_TRANSLATION[language][key]\r\nconst SMART_TRANSLATION = {\r\n  \"en\": {\r\n    \"pronous\": [\"my\", \"I\", \"me\", \"mine\", \"our\", \"ours\", \"us\", \"we\"],\r\n    \"prompt\": \"Based on your notes\",\r\n    \"initial_message\": \"Hi, I'm ChatGPT with access to your notes via Smart Connections. Ask me a question about your notes and I'll try to answer it.\",\r\n    \"try_placeholder\": `Try \"Based on my notes\" or \"Summarize [[this note]]\" or \"Important tasks in /folder/\"`\r\n  },\r\n  \"es\": {\r\n    \"pronous\": [\"mi\", \"yo\", \"m\u00ED\", \"t\u00FA\"],\r\n    \"prompt\": \"Bas\u00E1ndose en sus notas\",\r\n    \"initial_message\": \"Hola, soy ChatGPT con acceso a tus apuntes a trav\u00E9s de Smart Connections. Hazme una pregunta sobre tus apuntes e intentar\u00E9 responderte.\",\r\n    \"try_placeholder\": `Prueba \"Basado en mis notas\" o \"Resumen [[esta nota]]\" o \"Tareas importantes en /carpeta/\"`\r\n  },\r\n  \"fr\": {\r\n    \"pronous\": [\"me\", \"mon\", \"ma\", \"mes\", \"moi\", \"nous\", \"notre\", \"nos\", \"je\", \"j'\", \"m'\"],\r\n    \"prompt\": \"D'apr\u00E8s vos notes\",\r\n    \"initial_message\": \"Bonjour, je suis ChatGPT et j'ai acc\u00E8s \u00E0 vos notes via Smart Connections. Posez-moi une question sur vos notes et j'essaierai d'y r\u00E9pondre.\",\r\n    \"try_placeholder\": `Essayez \"D'apr\u00E8s mes notes\" ou \"R\u00E9sume [[cette note]]\" ou \"T\u00E2ches importantes dans /dossier/\"`\r\n  },\r\n  \"de\": {\r\n    \"pronous\": [\"mein\", \"meine\", \"meinen\", \"meiner\", \"meines\", \"mir\", \"uns\", \"unser\", \"unseren\", \"unserer\", \"unseres\"],\r\n    \"prompt\": \"Basierend auf Ihren Notizen\",\r\n    \"initial_message\": \"Hallo, ich bin ChatGPT und habe \u00FCber Smart Connections Zugang zu Ihren Notizen. Stellen Sie mir eine Frage zu Ihren Notizen und ich werde versuchen, sie zu beantworten.\",\r\n    \"try_placeholder\": `Versuchen Sie \"Basierend auf meinen Notizen\" oder \"Zusammenfassen [[dieser Notiz]]\" oder \"Wichtige Aufgaben im /Ordner/\"`\r\n  },\r\n  \"it\": {\r\n    \"pronous\": [\"mio\", \"mia\", \"miei\", \"mie\", \"noi\", \"nostro\", \"nostri\", \"nostra\", \"nostre\"],\r\n    \"prompt\": \"Sulla base degli appunti\",\r\n    \"initial_message\": \"Ciao, sono ChatGPT e ho accesso ai tuoi appunti tramite Smart Connections. Fatemi una domanda sui vostri appunti e cercher\u00F2 di rispondervi.\",\r\n    \"try_placeholder\": `Prova \"Sulla base dei miei appunti\" o \"Riassumi [[questo appunto]]\" o \"Compiti importanti in /cartella/\"`\r\n  },\r\n}\r\n\r\n// require built-in crypto module\r\nconst crypto = require(\"crypto\");\r\n// md5 hash using built in crypto module\r\nfunction md5(str) {\r\n  return crypto.createHash(\"md5\").update(str).digest(\"hex\");\r\n}\r\n\r\nclass SmartConnectionsPlugin extends Obsidian.Plugin {\r\n  // constructor\r\n  constructor() {\r\n    super(...arguments);\r\n    this.api = null;\r\n    this.embeddings_loaded = false;\r\n    this.file_exclusions = [];\r\n    this.folders = [];\r\n    this.has_new_embeddings = false;\r\n    this.header_exclusions = [];\r\n    this.nearest_cache = {};\r\n    this.path_only = [];\r\n    this.render_log = {};\r\n    this.render_log.deleted_embeddings = 0;\r\n    this.render_log.exclusions_logs = {};\r\n    this.render_log.failed_embeddings = [];\r\n    this.render_log.files = [];\r\n    this.render_log.new_embeddings = 0;\r\n    this.render_log.skipped_low_delta = {};\r\n    this.render_log.token_usage = 0;\r\n    this.render_log.tokens_saved_by_cache = 0;\r\n    this.retry_notice_timeout = null;\r\n    this.save_timeout = null;\r\n    this.sc_branding = {};\r\n    this.self_ref_kw_regex = null;\r\n    this.update_available = false;\r\n  }\r\n\r\n  async onload() {\r\n    // initialize when layout is ready\r\n    this.app.workspace.onLayoutReady(this.initialize.bind(this));\r\n  }\r\n  onunload() {\r\n    this.output_render_log();\r\n    console.log(\"unloading plugin\");\r\n    this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE);\r\n    this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE);\r\n  }\r\n  async initialize() {\r\n    console.log(\"Loading Smart Connections plugin\");\r\n    VERSION = this.manifest.version;\r\n    // VERSION = '1.0.0';\r\n    // console.log(VERSION);\r\n    await this.loadSettings();\r\n    // run after 3 seconds\r\n    setTimeout(this.check_for_update.bind(this), 3000);\r\n    // run check for update every 3 hours\r\n    setInterval(this.check_for_update.bind(this), 10800000);\r\n\r\n    this.addIcon();\r\n    this.addCommand({\r\n      id: \"sc-find-notes\",\r\n      name: \"Find: Make Smart Connections\",\r\n      icon: \"pencil_icon\",\r\n      hotkeys: [],\r\n      // editorCallback: async (editor) => {\r\n      editorCallback: async (editor) => {\r\n        if(editor.somethingSelected()) {\r\n          // get selected text\r\n          let selected_text = editor.getSelection();\r\n          // render connections from selected text\r\n          await this.make_connections(selected_text);\r\n        } else {\r\n          // clear nearest_cache on manual call to make connections\r\n          this.nearest_cache = {};\r\n          // console.log(\"Cleared nearest_cache\");\r\n          await this.make_connections();\r\n        }\r\n      }\r\n    });\r\n    this.addCommand({\r\n      id: \"smart-connections-view\",\r\n      name: \"Open: View Smart Connections\",\r\n      callback: () => {\r\n        this.open_view();\r\n      }\r\n    });\r\n    // open chat command\r\n    this.addCommand({\r\n      id: \"smart-connections-chat\",\r\n      name: \"Open: Smart Chat Conversation\",\r\n      callback: () => {\r\n        this.open_chat();\r\n      }\r\n    });\r\n    // open random note from nearest cache\r\n    this.addCommand({\r\n      id: \"smart-connections-random\",\r\n      name: \"Open: Random Note from Smart Connections\",\r\n      callback: () => {\r\n        this.open_random_note();\r\n      }\r\n    });\r\n    // add settings tab\r\n    this.addSettingTab(new SmartConnectionsSettingsTab(this.app, this));\r\n    // register main view type\r\n    this.registerView(SMART_CONNECTIONS_VIEW_TYPE, (leaf) => (new SmartConnectionsView(leaf, this)));\r\n    // register chat view type\r\n    this.registerView(SMART_CONNECTIONS_CHAT_VIEW_TYPE, (leaf) => (new SmartConnectionsChatView(leaf, this)));\r\n    // code-block renderer\r\n    this.registerMarkdownCodeBlockProcessor(\"smart-connections\", this.render_code_block.bind(this));\r\n\r\n    // if this settings.view_open is true, open view on startup\r\n    if(this.settings.view_open) {\r\n      this.open_view();\r\n    }\r\n    // if this settings.chat_open is true, open chat on startup\r\n    if(this.settings.chat_open) {\r\n      this.open_chat();\r\n    }\r\n    // on new version\r\n    if(this.settings.version !== VERSION) {\r\n      // update version\r\n      this.settings.version = VERSION;\r\n      // save settings\r\n      await this.saveSettings();\r\n      // open view\r\n      this.open_view();\r\n    }\r\n    // check github release endpoint if update is available\r\n    this.add_to_gitignore();\r\n    /**\r\n     * EXPERIMENTAL\r\n     * - window-based API access\r\n     * - code-block rendering\r\n     */\r\n    this.api = new ScSearchApi(this.app, this);\r\n    // register API to global window object\r\n    (window[\"SmartSearchApi\"] = this.api) && this.register(() => delete window[\"SmartSearchApi\"]);\r\n\r\n  }\r\n\r\n  async init_vecs() {\r\n    this.smart_vec_lite = new VecLite({\r\n      folder_path: \".smart-connections\",\r\n      exists_adapter: this.app.vault.adapter.exists.bind(this.app.vault.adapter),\r\n      mkdir_adapter: this.app.vault.adapter.mkdir.bind(this.app.vault.adapter),\r\n      read_adapter: this.app.vault.adapter.read.bind(this.app.vault.adapter),\r\n      rename_adapter: this.app.vault.adapter.rename.bind(this.app.vault.adapter),\r\n      stat_adapter: this.app.vault.adapter.stat.bind(this.app.vault.adapter),\r\n      write_adapter: this.app.vault.adapter.write.bind(this.app.vault.adapter),\r\n    });\r\n    this.embeddings_loaded = await this.smart_vec_lite.load();\r\n    return this.embeddings_loaded;\r\n  }\r\n  async update_to_v2() {\r\n    // if license key is not set, return\r\n    if(!this.settings.license_key) return new Obsidian.Notice(\"[Smart Connections] Supporter license key required for early access to V2\");\r\n    // download https://github.com/brianpetro/obsidian-smart-connections/releases/download/1.6.37/main.js\r\n    const v2 = await (0, Obsidian.requestUrl)({\r\n      url: \"https://sync.smartconnections.app/download_v2\",\r\n      method: \"POST\",\r\n      headers: {\r\n        \"Content-Type\": \"application/json\",\r\n      },\r\n      body: JSON.stringify({\r\n        license_key: this.settings.license_key,\r\n      })\r\n    });\r\n    if(v2.status !== 200) return console.error(\"Error downloading version 2\", v2);\r\n    console.log(v2);\r\n    await this.app.vault.adapter.write(\".obsidian/plugins/smart-connections/main.js\", v2.json.main); // add new\r\n    await this.app.vault.adapter.write(\".obsidian/plugins/smart-connections/manifest.json\", v2.json.manifest); // add new\r\n    await this.app.vault.adapter.write(\".obsidian/plugins/smart-connections/styles.css\", v2.json.styles); // add new\r\n    window.restart_plugin = async (id) => {\r\n      console.log(\"restarting plugin\", id);\r\n      await window.app.plugins.disablePlugin(id);\r\n      await window.app.plugins.enablePlugin(id);\r\n      console.log(\"plugin restarted\", id);\r\n    }\r\n    window.restart_plugin(this.manifest.id);\r\n  }\r\n\r\n  async loadSettings() {\r\n    this.settings = Object.assign({}, DEFAULT_SETTINGS, await this.loadData());\r\n    // load file exclusions if not blank\r\n    if(this.settings.file_exclusions && this.settings.file_exclusions.length > 0) {\r\n      // split file exclusions into array and trim whitespace\r\n      this.file_exclusions = this.settings.file_exclusions.split(/[,\\n\\s]/)\r\n        .filter((file) => file.length > 0)\r\n        .map((file) => {\r\n        return file.trim();\r\n      });\r\n    }\r\n    // load folder exclusions if not blank\r\n    if(this.settings.folder_exclusions && this.settings.folder_exclusions.length > 0) {\r\n      // add slash to end of folder name if not present\r\n      const folder_exclusions = this.settings.folder_exclusions\r\n        .split(/[,\\n\\s]/)\r\n        .filter((file) => file.length > 0)\r\n        .map((folder) => {\r\n          folder = folder.trim();\r\n          return folder.slice(-1) === \"/\" ? folder : `${folder}/`;\r\n        });\r\n      // merge folder exclusions with file exclusions\r\n      this.file_exclusions = this.file_exclusions.concat(folder_exclusions);\r\n    }\r\n    // load header exclusions if not blank\r\n    if(this.settings.header_exclusions && this.settings.header_exclusions.length > 0) {\r\n      this.header_exclusions = this.settings.header_exclusions\r\n        .split(/[\\s\\n,]/)\r\n        .filter((file) => file.length > 0)\r\n        .map((header) => {\r\n          return header.trim();\r\n        });\r\n    }\r\n    // load path_only if not blank\r\n    if(this.settings.path_only && this.settings.path_only.length > 0) {\r\n      this.path_only = this.settings.path_only\r\n        .split(/[\\s\\n,]/)\r\n        .filter((file) => file.length > 0)\r\n        .map((path) => {\r\n          return path.trim();\r\n        });\r\n    }\r\n    // load self_ref_kw_regex\r\n    this.self_ref_kw_regex = new RegExp(`\\\\b(${SMART_TRANSLATION[this.settings.language].pronous.join(\"|\")})\\\\b`, \"gi\");\r\n    // load failed files\r\n    await this.load_failed_files();\r\n  }\r\n  async saveSettings(rerender=false) {\r\n    await this.saveData(this.settings);\r\n    // re-load settings into memory\r\n    await this.loadSettings();\r\n    // re-render view if set to true (for example, after adding API key)\r\n    if(rerender) {\r\n      this.nearest_cache = {};\r\n      await this.make_connections();\r\n    }\r\n  }\r\n\r\n  // check for update\r\n  async check_for_update() {\r\n    // fail silently, ex. if no internet connection\r\n    try {\r\n      // get latest release version from github\r\n      const response = await (0, Obsidian.requestUrl)({\r\n        url: \"https://api.github.com/repos/brianpetro/obsidian-smart-connections/releases/latest\",\r\n        method: \"GET\",\r\n        headers: {\r\n          \"Content-Type\": \"application/json\",\r\n        },\r\n        contentType: \"application/json\",\r\n      });\r\n      // get version number from response\r\n      const latest_release = JSON.parse(response.text).tag_name;\r\n      // console.log(`Latest release: ${latest_release}`);\r\n      // if latest_release is newer than current version, show message\r\n      if(latest_release !== VERSION) {\r\n        new Obsidian.Notice(`[Smart Connections] A new version is available! (v${latest_release})`);\r\n        this.update_available = true;\r\n        this.render_brand(\"all\")\r\n      }\r\n    } catch (error) {\r\n      console.log(error);\r\n    }\r\n  }\r\n\r\n  async render_code_block(contents, container, ctx) {\r\n    let nearest;\r\n    if(contents.trim().length > 0) {\r\n      nearest = await this.api.search(contents);\r\n    } else {\r\n      // use ctx to get file\r\n      console.log(ctx);\r\n      const file = this.app.vault.getAbstractFileByPath(ctx.sourcePath);\r\n      nearest = await this.find_note_connections(file);\r\n    }\r\n    if (nearest.length) {\r\n      this.update_results(container, nearest);\r\n    }\r\n  }\r\n\r\n  async make_connections(selected_text=null) {\r\n    let view = this.get_view();\r\n    if (!view) {\r\n      // open view if not open\r\n      await this.open_view();\r\n      view = this.get_view();\r\n    }\r\n    await view.render_connections(selected_text);\r\n  }\r\n\r\n  addIcon(){\r\n    Obsidian.addIcon(\"smart-connections\", `<path d=\"M50,20 L80,40 L80,60 L50,100\" stroke=\"currentColor\" stroke-width=\"4\" fill=\"none\"/>\r\n    <path d=\"M30,50 L55,70\" stroke=\"currentColor\" stroke-width=\"5\" fill=\"none\"/>\r\n    <circle cx=\"50\" cy=\"20\" r=\"9\" fill=\"currentColor\"/>\r\n    <circle cx=\"80\" cy=\"40\" r=\"9\" fill=\"currentColor\"/>\r\n    <circle cx=\"80\" cy=\"70\" r=\"9\" fill=\"currentColor\"/>\r\n    <circle cx=\"50\" cy=\"100\" r=\"9\" fill=\"currentColor\"/>\r\n    <circle cx=\"30\" cy=\"50\" r=\"9\" fill=\"currentColor\"/>`);\r\n  }\r\n\r\n  // open random note\r\n  async open_random_note() {\r\n    const curr_file = this.app.workspace.getActiveFile();\r\n    const curr_key = md5(curr_file.path);\r\n    // if no nearest cache, create Obsidian notice\r\n    if(typeof this.nearest_cache[curr_key] === \"undefined\") {\r\n      new Obsidian.Notice(\"[Smart Connections] No Smart Connections found. Open a note to get Smart Connections.\");\r\n      return;\r\n    }\r\n    // get random from nearest cache\r\n    const rand = Math.floor(Math.random() * this.nearest_cache[curr_key].length/2); // divide by 2 to limit to top half of results\r\n    const random_file = this.nearest_cache[curr_key][rand];\r\n    // open random file\r\n    this.open_note(random_file);\r\n  }\r\n\r\n  async open_view() {\r\n    if(this.get_view()){\r\n      console.log(\"Smart Connections view already open\");\r\n      return;\r\n    }\r\n    this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE);\r\n    await this.app.workspace.getRightLeaf(false).setViewState({\r\n      type: SMART_CONNECTIONS_VIEW_TYPE,\r\n      active: true,\r\n    });\r\n    this.app.workspace.revealLeaf(\r\n      this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE)[0]\r\n    );\r\n  }\r\n  // source: https://github.com/obsidianmd/obsidian-releases/blob/master/plugin-review.md#avoid-managing-references-to-custom-views\r\n  get_view() {\r\n    for (let leaf of this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE)) {\r\n      if (leaf.view instanceof SmartConnectionsView) {\r\n        return leaf.view;\r\n      }\r\n    }\r\n  }\r\n  // open chat view\r\n  async open_chat(retries=0) {\r\n    if(!this.embeddings_loaded) {\r\n      console.log(\"embeddings not loaded yet\");\r\n      if(retries < 3) {\r\n        // wait and try again\r\n        setTimeout(() => {\r\n          this.open_chat(retries+1);\r\n        }, 1000 * (retries+1));\r\n        return;\r\n      }\r\n      console.log(\"embeddings still not loaded, opening smart view\");\r\n      this.open_view();\r\n      return;\r\n    }\r\n    this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE);\r\n    if (!this.settings.open_in_big_view) {\r\n      await this.app.workspace.getRightLeaf(false).setViewState({\r\n        type: SMART_CONNECTIONS_CHAT_VIEW_TYPE,\r\n        active: true,\r\n      });\r\n    } else {\r\n      await this.app.workspace.getLeaf(true).setViewState({\r\n        type: SMART_CONNECTIONS_CHAT_VIEW_TYPE,\r\n        active: true,\r\n      })\r\n    }\r\n    this.app.workspace.revealLeaf(\r\n      this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE)[0]\r\n    );\r\n  }\r\n\r\n  // get embeddings for all files\r\n  async get_all_embeddings() {\r\n    // get all files in vault and filter all but markdown and canvas files\r\n    const files = (await this.app.vault.getFiles()).filter((file) => file instanceof Obsidian.TFile && (file.extension === \"md\" || file.extension === \"canvas\"));\r\n    // const files = await this.app.vault.getMarkdownFiles();\r\n    // get open files to skip if file is currently open\r\n    const open_files = this.app.workspace.getLeavesOfType(\"markdown\").map((leaf) => leaf.view.file);\r\n    const clean_up_log = this.smart_vec_lite.clean_up_embeddings(files);\r\n    if(this.settings.log_render){\r\n      this.render_log.total_files = files.length;\r\n      this.render_log.deleted_embeddings = clean_up_log.deleted_embeddings;\r\n      this.render_log.total_embeddings = clean_up_log.total_embeddings;\r\n    }\r\n    // batch embeddings\r\n    let batch_promises = [];\r\n    for (let i = 0; i < files.length; i++) {\r\n      // skip if path contains a #\r\n      if(files[i].path.indexOf(\"#\") > -1) {\r\n        // console.log(\"skipping file '\"+files[i].path+\"' (path contains #)\");\r\n        this.log_exclusion(\"path contains #\");\r\n        continue;\r\n      }\r\n      // skip if file already has embedding and embedding.mtime is greater than or equal to file.mtime\r\n      if(this.smart_vec_lite.mtime_is_current(md5(files[i].path), files[i].stat.mtime)) {\r\n        // log skipping file\r\n        // console.log(\"skipping file (mtime)\");\r\n        continue;\r\n      }\r\n      // check if file is in failed_files\r\n      if(this.settings.failed_files.indexOf(files[i].path) > -1) {\r\n        // log skipping file\r\n        // console.log(\"skipping previously failed file, use button in settings to retry\");\r\n        // use setTimeout to prevent multiple notices\r\n        if(this.retry_notice_timeout) {\r\n          clearTimeout(this.retry_notice_timeout);\r\n          this.retry_notice_timeout = null;\r\n        }\r\n        // limit to one notice every 10 minutes\r\n        if(!this.recently_sent_retry_notice){\r\n          new Obsidian.Notice(\"Smart Connections: Skipping previously failed file, use button in settings to retry\");\r\n          this.recently_sent_retry_notice = true;\r\n          setTimeout(() => {\r\n            this.recently_sent_retry_notice = false;\r\n          }, 600000);\r\n        }\r\n        continue;\r\n      }\r\n      // skip files where path contains any exclusions\r\n      let skip = false;\r\n      for (const fileExclusion of this.file_exclusions) {\r\n        if(files[i].path.includes(fileExclusion)) {\r\n          skip = true;\r\n          this.log_exclusion(fileExclusion);\r\n          // break out of loop\r\n          break;\r\n        }\r\n      }\r\n      if(skip) {\r\n        continue; // to next file\r\n      }\r\n      // check if file is open\r\n      if(open_files.indexOf(files[i]) > -1) {\r\n        // console.log(\"skipping file (open)\");\r\n        continue;\r\n      }\r\n      try {\r\n        // push promise to batch_promises\r\n        batch_promises.push(this.get_file_embeddings(files[i], false));\r\n      } catch (error) {\r\n        console.log(error);\r\n      }\r\n      // if batch_promises length is 10\r\n      if(batch_promises.length > 3) {\r\n        // wait for all promises to resolve\r\n        await Promise.all(batch_promises);\r\n        // clear batch_promises\r\n        batch_promises = [];\r\n      }\r\n\r\n      // save embeddings JSON to file every 100 files to save progress on bulk embedding\r\n      if(i > 0 && i % 100 === 0) {\r\n        await this.save_embeddings_to_file();\r\n      }\r\n    }\r\n    // wait for all promises to resolve\r\n    await Promise.all(batch_promises);\r\n    // write embeddings JSON to file\r\n    await this.save_embeddings_to_file();\r\n    // if render_log.failed_embeddings then update failed_embeddings.txt\r\n    if(this.render_log.failed_embeddings.length > 0) {\r\n      await this.save_failed_embeddings();\r\n    }\r\n  }\r\n\r\n  async save_embeddings_to_file(force=false) {\r\n    if(!this.has_new_embeddings){\r\n      return;\r\n    }\r\n    // console.log(\"new embeddings, saving to file\");\r\n    if(!force) {\r\n      // prevent excessive writes to embeddings file by waiting 1 minute before writing\r\n      if(this.save_timeout) {\r\n        clearTimeout(this.save_timeout);\r\n        this.save_timeout = null;\r\n      }\r\n      this.save_timeout = setTimeout(() => {\r\n        // console.log(\"writing embeddings to file\");\r\n        this.save_embeddings_to_file(true);\r\n        // clear timeout\r\n        if(this.save_timeout) {\r\n          clearTimeout(this.save_timeout);\r\n          this.save_timeout = null;\r\n        }\r\n      }, 30000);\r\n      console.log(\"scheduled save\");\r\n      return;\r\n    }\r\n\r\n    try{\r\n      // use smart_vec_lite\r\n      await this.smart_vec_lite.save();\r\n      this.has_new_embeddings = false;\r\n    }catch(error){\r\n      console.log(error);\r\n      new Obsidian.Notice(\"Smart Connections: \"+error.message);\r\n    }\r\n\r\n  }\r\n  // save failed embeddings to file from render_log.failed_embeddings\r\n  async save_failed_embeddings () {\r\n    // write failed_embeddings to file one line per failed embedding\r\n    let failed_embeddings = [];\r\n    // if file already exists then read it\r\n    const failed_embeddings_file_exists = await this.app.vault.adapter.exists(\".smart-connections/failed-embeddings.txt\");\r\n    if(failed_embeddings_file_exists) {\r\n      failed_embeddings = await this.app.vault.adapter.read(\".smart-connections/failed-embeddings.txt\");\r\n      // split failed_embeddings into array\r\n      failed_embeddings = failed_embeddings.split(\"\\r\\n\");\r\n    }\r\n    // merge failed_embeddings with render_log.failed_embeddings\r\n    failed_embeddings = failed_embeddings.concat(this.render_log.failed_embeddings);\r\n    // remove duplicates\r\n    failed_embeddings = [...new Set(failed_embeddings)];\r\n    // sort failed_embeddings array alphabetically\r\n    failed_embeddings.sort();\r\n    // convert failed_embeddings array to string\r\n    failed_embeddings = failed_embeddings.join(\"\\r\\n\");\r\n    // write failed_embeddings to file\r\n    await this.app.vault.adapter.write(\".smart-connections/failed-embeddings.txt\", failed_embeddings);\r\n    // reload failed_embeddings to prevent retrying failed files until explicitly requested\r\n    await this.load_failed_files();\r\n  }\r\n\r\n  // load failed files from failed-embeddings.txt\r\n  async load_failed_files () {\r\n    // check if failed-embeddings.txt exists\r\n    const failed_embeddings_file_exists = await this.app.vault.adapter.exists(\".smart-connections/failed-embeddings.txt\");\r\n    if(!failed_embeddings_file_exists) {\r\n      this.settings.failed_files = [];\r\n      console.log(\"No failed files.\");\r\n      return;\r\n    }\r\n    // read failed-embeddings.txt\r\n    const failed_embeddings = await this.app.vault.adapter.read(\".smart-connections/failed-embeddings.txt\");\r\n    // split failed_embeddings into array and remove empty lines\r\n    const failed_embeddings_array = failed_embeddings.split(\"\\r\\n\");\r\n    // split at '#' and reduce into unique file paths\r\n    const failed_files = failed_embeddings_array.map(embedding => embedding.split(\"#\")[0]).reduce((unique, item) => unique.includes(item) ? unique : [...unique, item], []);\r\n    // return failed_files\r\n    this.settings.failed_files = failed_files;\r\n    // console.log(failed_files);\r\n  }\r\n  // retry failed embeddings\r\n  async retry_failed_files () {\r\n    // remove failed files from failed_files\r\n    this.settings.failed_files = [];\r\n    // if failed-embeddings.txt exists then delete it\r\n    const failed_embeddings_file_exists = await this.app.vault.adapter.exists(\".smart-connections/failed-embeddings.txt\");\r\n    if(failed_embeddings_file_exists) {\r\n      await this.app.vault.adapter.remove(\".smart-connections/failed-embeddings.txt\");\r\n    }\r\n    // run get all embeddings\r\n    await this.get_all_embeddings();\r\n  }\r\n\r\n\r\n  // add .smart-connections to .gitignore to prevent issues with large, frequently updated embeddings file(s)\r\n  async add_to_gitignore() {\r\n    if(!(await this.app.vault.adapter.exists(\".gitignore\"))) {\r\n      return; // if .gitignore doesn't exist then don't add .smart-connections to .gitignore\r\n    }\r\n    let gitignore_file = await this.app.vault.adapter.read(\".gitignore\");\r\n    // if .smart-connections not in .gitignore\r\n    if (gitignore_file.indexOf(\".smart-connections\") < 0) {\r\n      // add .smart-connections to .gitignore\r\n      let add_to_gitignore = \"\\n\\n# Ignore Smart Connections folder because embeddings file is large and updated frequently\";\r\n      add_to_gitignore += \"\\n.smart-connections\";\r\n      await this.app.vault.adapter.write(\".gitignore\", gitignore_file + add_to_gitignore);\r\n      console.log(\"added .smart-connections to .gitignore\");\r\n    }\r\n  }\r\n\r\n  // force refresh embeddings file but first rename existing embeddings file to .smart-connections/embeddings-YYYY-MM-DD.json\r\n  async force_refresh_embeddings_file() {\r\n    new Obsidian.Notice(\"Smart Connections: embeddings file Force Refreshed, making new connections...\");\r\n    // force refresh\r\n    await this.smart_vec_lite.force_refresh();\r\n    // trigger making new connections\r\n    await this.get_all_embeddings();\r\n    this.output_render_log();\r\n    new Obsidian.Notice(\"Smart Connections: embeddings file Force Refreshed, new connections made.\");\r\n  }\r\n\r\n  // get embeddings for embed_input\r\n  async get_file_embeddings(curr_file, save=true) {\r\n    // let batch_promises = [];\r\n    let req_batch = [];\r\n    let blocks = [];\r\n    // initiate curr_file_key from md5(curr_file.path)\r\n    const curr_file_key = md5(curr_file.path);\r\n    // intiate file_file_embed_input by removing .md and converting file path to breadcrumbs (\" > \")\r\n    let file_embed_input = curr_file.path.replace(\".md\", \"\");\r\n    file_embed_input = file_embed_input.replace(/\\//g, \" > \");\r\n    // embed on file.name/title only if path_only path matcher specified in settings\r\n    let path_only = false;\r\n    for(let j = 0; j < this.path_only.length; j++) {\r\n      if(curr_file.path.indexOf(this.path_only[j]) > -1) {\r\n        path_only = true;\r\n        console.log(\"title only file with matcher: \" + this.path_only[j]);\r\n        // break out of loop\r\n        break;\r\n      }\r\n    }\r\n    // return early if path_only\r\n    if(path_only) {\r\n      req_batch.push([curr_file_key, file_embed_input, {\r\n        mtime: curr_file.stat.mtime,\r\n        path: curr_file.path,\r\n      }]);\r\n      await this.get_embeddings_batch(req_batch);\r\n      return;\r\n    }\r\n    /**\r\n     * BEGIN Canvas file type Embedding\r\n     */\r\n    if(curr_file.extension === \"canvas\") {\r\n      // get file contents and parse as JSON\r\n      const canvas_contents = await this.app.vault.cachedRead(curr_file);\r\n      if((typeof canvas_contents === \"string\") && (canvas_contents.indexOf(\"nodes\") > -1)) {\r\n        const canvas_json = JSON.parse(canvas_contents);\r\n        // for each object in nodes array\r\n        for(let j = 0; j < canvas_json.nodes.length; j++) {\r\n          // if object has text property\r\n          if(canvas_json.nodes[j].text) {\r\n            // add to file_embed_input\r\n            file_embed_input += \"\\n\" + canvas_json.nodes[j].text;\r\n          }\r\n          // if object has file property\r\n          if(canvas_json.nodes[j].file) {\r\n            // add to file_embed_input\r\n            file_embed_input += \"\\nLink: \" + canvas_json.nodes[j].file;\r\n          }\r\n        }\r\n      }\r\n      // console.log(file_embed_input);\r\n      req_batch.push([curr_file_key, file_embed_input, {\r\n        mtime: curr_file.stat.mtime,\r\n        path: curr_file.path,\r\n      }]);\r\n      await this.get_embeddings_batch(req_batch);\r\n      return;\r\n    }\r\n\r\n    /**\r\n     * BEGIN Block \"section\" embedding\r\n     */\r\n    // get file contents\r\n    const note_contents = await this.app.vault.cachedRead(curr_file);\r\n    let processed_since_last_save = 0;\r\n    const note_sections = this.block_parser(note_contents, curr_file.path);\r\n    // console.log(note_sections);\r\n    // if note has more than one section (if only one then its same as full-content)\r\n    if(note_sections.length > 1) {\r\n      // for each section in file\r\n      //console.log(\"Sections: \" + note_sections.length);\r\n      for (let j = 0; j < note_sections.length; j++) {\r\n        // get embed_input for block\r\n        const block_embed_input = note_sections[j].text;\r\n        // console.log(note_sections[j].path);\r\n        // get block key from block.path (contains both file.path and header path)\r\n        const block_key = md5(note_sections[j].path);\r\n        blocks.push(block_key);\r\n        // skip if length of block_embed_input same as length of embeddings[block_key].meta.size\r\n        // TODO consider rounding to nearest 10 or 100 for fuzzy matching\r\n        if (this.smart_vec_lite.get_size(block_key) === block_embed_input.length) {\r\n          // log skipping file\r\n          // console.log(\"skipping block (len)\");\r\n          continue;\r\n        }\r\n        // add hash to blocks to prevent empty blocks triggering full-file embedding\r\n        // skip if embeddings key already exists and block mtime is greater than or equal to file mtime\r\n        if(this.smart_vec_lite.mtime_is_current(block_key, curr_file.stat.mtime)) {\r\n          // log skipping file\r\n          // console.log(\"skipping block (mtime)\");\r\n          continue;\r\n        }\r\n        // skip if hash is present in embeddings and hash of block_embed_input is equal to hash in embeddings\r\n        const block_hash = md5(block_embed_input.trim());\r\n        if(this.smart_vec_lite.get_hash(block_key) === block_hash) {\r\n          // log skipping file\r\n          // console.log(\"skipping block (hash)\");\r\n          continue;\r\n        }\r\n\r\n        // create req_batch for batching requests\r\n        req_batch.push([block_key, block_embed_input, {\r\n          // oldmtime: curr_file.stat.mtime,\r\n          // get current datetime as unix timestamp\r\n          mtime: Date.now(),\r\n          hash: block_hash,\r\n          parent: curr_file_key,\r\n          path: note_sections[j].path,\r\n          size: block_embed_input.length,\r\n        }]);\r\n        if(req_batch.length > 9) {\r\n          // add batch to batch_promises\r\n          await this.get_embeddings_batch(req_batch);\r\n          processed_since_last_save += req_batch.length;\r\n          // log embedding\r\n          // console.log(\"embedding: \" + curr_file.path);\r\n          if (processed_since_last_save >= 30) {\r\n            // write embeddings JSON to file\r\n            await this.save_embeddings_to_file();\r\n            // reset processed_since_last_save\r\n            processed_since_last_save = 0;\r\n          }\r\n          // reset req_batch\r\n          req_batch = [];\r\n        }\r\n      }\r\n    }\r\n    // if req_batch is not empty\r\n    if(req_batch.length > 0) {\r\n      // process remaining req_batch\r\n      await this.get_embeddings_batch(req_batch);\r\n      req_batch = [];\r\n      processed_since_last_save += req_batch.length;\r\n    }\r\n\r\n    /**\r\n     * BEGIN File \"full note\" embedding\r\n     */\r\n\r\n    // if file length is less than ~8000 tokens use full file contents\r\n    // else if file length is greater than 8000 tokens build file_embed_input from file headings\r\n    file_embed_input += `:\\n`;\r\n    /**\r\n     * TODO: improve/refactor the following \"large file reduce to headings\" logic\r\n     */\r\n    if(note_contents.length < MAX_EMBED_STRING_LENGTH) {\r\n      file_embed_input += note_contents\r\n    }else{\r\n      const note_meta_cache = this.app.metadataCache.getFileCache(curr_file);\r\n      // for each heading in file\r\n      if(typeof note_meta_cache.headings === \"undefined\") {\r\n        // console.log(\"no headings found, using first chunk of file instead\");\r\n        file_embed_input += note_contents.substring(0, MAX_EMBED_STRING_LENGTH);\r\n      }else{\r\n        let note_headings = \"\";\r\n        for (let j = 0; j < note_meta_cache.headings.length; j++) {\r\n          // get heading level\r\n          const heading_level = note_meta_cache.headings[j].level;\r\n          // get heading text\r\n          const heading_text = note_meta_cache.headings[j].heading;\r\n          // build markdown heading\r\n          let md_heading = \"\";\r\n          for (let k = 0; k < heading_level; k++) {\r\n            md_heading += \"#\";\r\n          }\r\n          // add heading to note_headings\r\n          note_headings += `${md_heading} ${heading_text}\\n`;\r\n        }\r\n        //console.log(note_headings);\r\n        file_embed_input += note_headings\r\n        if(file_embed_input.length > MAX_EMBED_STRING_LENGTH) {\r\n          file_embed_input = file_embed_input.substring(0, MAX_EMBED_STRING_LENGTH);\r\n        }\r\n      }\r\n    }\r\n    // skip embedding full file if blocks is not empty and all hashes are present in embeddings\r\n    // better than hashing file_embed_input because more resilient to inconsequential changes (whitespace between headings)\r\n    const file_hash = md5(file_embed_input.trim());\r\n    const existing_hash = this.smart_vec_lite.get_hash(curr_file_key);\r\n    if(existing_hash && (file_hash === existing_hash)) {\r\n      // console.log(\"skipping file (hash): \" + curr_file.path);\r\n      this.update_render_log(blocks, file_embed_input);\r\n      return;\r\n    };\r\n\r\n    // if not already skipping and blocks are present\r\n    const existing_blocks = this.smart_vec_lite.get_children(curr_file_key);\r\n    let existing_has_all_blocks = true;\r\n    if(existing_blocks && Array.isArray(existing_blocks) && (blocks.length > 0)) {\r\n      // if all blocks are in existing_blocks then skip (allows deletion of small blocks without triggering full file embedding)\r\n      for (let j = 0; j < blocks.length; j++) {\r\n        if(existing_blocks.indexOf(blocks[j]) === -1) {\r\n          existing_has_all_blocks = false;\r\n          break;\r\n        }\r\n      }\r\n    }\r\n    // if existing has all blocks then check file size for delta\r\n    if(existing_has_all_blocks){\r\n      // get current note file size\r\n      const curr_file_size = curr_file.stat.size;\r\n      // get file size from embeddings\r\n      const prev_file_size = this.smart_vec_lite.get_size(curr_file_key);\r\n      if (prev_file_size) {\r\n        // if curr file size is less than 10% different from prev file size\r\n        const file_delta_pct = Math.round((Math.abs(curr_file_size - prev_file_size) / curr_file_size) * 100);\r\n        if(file_delta_pct < 10) {\r\n          // skip embedding\r\n          // console.log(\"skipping file (size) \" + curr_file.path);\r\n          this.render_log.skipped_low_delta[curr_file.name] = file_delta_pct + \"%\";\r\n          this.update_render_log(blocks, file_embed_input);\r\n          return;\r\n        }\r\n      }\r\n    }\r\n    let meta = {\r\n      mtime: curr_file.stat.mtime,\r\n      hash: file_hash,\r\n      path: curr_file.path,\r\n      size: curr_file.stat.size,\r\n      children: blocks,\r\n    };\r\n    // batch_promises.push(this.get_embeddings(curr_file_key, file_embed_input, meta));\r\n    req_batch.push([curr_file_key, file_embed_input, meta]);\r\n    // send batch request\r\n    await this.get_embeddings_batch(req_batch);\r\n\r\n    // log embedding\r\n    // console.log(\"embedding: \" + curr_file.path);\r\n    if (save) {\r\n      // write embeddings JSON to file\r\n      await this.save_embeddings_to_file();\r\n    }\r\n\r\n  }\r\n\r\n  update_render_log(blocks, file_embed_input) {\r\n    if (blocks.length > 0) {\r\n      // multiply by 2 because implies we saved token spending on blocks(sections), too\r\n      this.render_log.tokens_saved_by_cache += file_embed_input.length / 2;\r\n    } else {\r\n      // calc tokens saved by cache: divide by 4 for token estimate\r\n      this.render_log.tokens_saved_by_cache += file_embed_input.length / 4;\r\n    }\r\n  }\r\n\r\n  async get_embeddings_batch(req_batch) {\r\n    console.log(\"get_embeddings_batch\");\r\n    // if req_batch is empty then return\r\n    if(req_batch.length === 0) return;\r\n    // create arrary of embed_inputs from req_batch[i][1]\r\n    const embed_inputs = req_batch.map((req) => req[1]);\r\n    // request embeddings from embed_inputs\r\n    const requestResults = await this.request_embedding_from_input(embed_inputs);\r\n    // if requestResults is null then return\r\n    if(!requestResults) {\r\n      console.log(\"failed embedding batch\");\r\n      // log failed file names to render_log\r\n      this.render_log.failed_embeddings = [...this.render_log.failed_embeddings, ...req_batch.map((req) => req[2].path)];\r\n      return;\r\n    }\r\n    // if requestResults is not null\r\n    if(requestResults){\r\n      this.has_new_embeddings = true;\r\n      // add embedding key to render_log\r\n      if(this.settings.log_render){\r\n        if(this.settings.log_render_files){\r\n          this.render_log.files = [...this.render_log.files, ...req_batch.map((req) => req[2].path)];\r\n        }\r\n        this.render_log.new_embeddings += req_batch.length;\r\n        // add token usage to render_log\r\n        this.render_log.token_usage += requestResults.usage.total_tokens;\r\n      }\r\n      // console.log(requestResults.data.length);\r\n      // loop through requestResults.data\r\n      for(let i = 0; i < requestResults.data.length; i++) {\r\n        const vec = requestResults.data[i].embedding;\r\n        const index = requestResults.data[i].index;\r\n        if(vec) {\r\n          const key = req_batch[index][0];\r\n          const meta = req_batch[index][2];\r\n          this.smart_vec_lite.save_embedding(key, vec, meta);\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  async request_embedding_from_input(embed_input, retries = 0) {\r\n    // (FOR TESTING) test fail process by forcing fail\r\n    // return null;\r\n    // check if embed_input is a string\r\n    // if(typeof embed_input !== \"string\") {\r\n    //   console.log(\"embed_input is not a string\");\r\n    //   return null;\r\n    // }\r\n    // check if embed_input is empty\r\n    if(embed_input.length === 0) {\r\n      console.log(\"embed_input is empty\");\r\n      return null;\r\n    }\r\n    const usedParams = {\r\n      model: \"text-embedding-ada-002\",\r\n      input: embed_input,\r\n    };\r\n    // console.log(this.settings.api_key);\r\n    const reqParams = {\r\n      url: `https://api.openai.com/v1/embeddings`,\r\n      method: \"POST\",\r\n      body: JSON.stringify(usedParams),\r\n      headers: {\r\n        \"Content-Type\": \"application/json\",\r\n        \"Authorization\": `Bearer ${this.settings.api_key}`\r\n      }\r\n    };\r\n    let resp;\r\n    try {\r\n      resp = await (0, Obsidian.request)(reqParams)\r\n      return JSON.parse(resp);\r\n    } catch (error) {\r\n      // retry request if error is 429\r\n      if((error.status === 429) && (retries < 3)) {\r\n        retries++;\r\n        // exponential backoff\r\n        const backoff = Math.pow(retries, 2);\r\n        console.log(`retrying request (429) in ${backoff} seconds...`);\r\n        await new Promise(r => setTimeout(r, 1000 * backoff));\r\n        return await this.request_embedding_from_input(embed_input, retries);\r\n      }\r\n      // log full error to console\r\n      console.log(resp);\r\n      // console.log(\"first line of embed: \" + embed_input.substring(0, embed_input.indexOf(\"\\n\")));\r\n      // console.log(\"embed input length: \"+ embed_input.length);\r\n      // if(Array.isArray(embed_input)) {\r\n      //   console.log(embed_input.map((input) => input.length));\r\n      // }\r\n      // console.log(\"erroneous embed input: \" + embed_input);\r\n      console.log(error);\r\n      // console.log(usedParams);\r\n      // console.log(usedParams.input.length);\r\n      return null;\r\n    }\r\n  }\r\n  async test_api_key() {\r\n    const embed_input = \"This is a test of the OpenAI API.\";\r\n    const resp = await this.request_embedding_from_input(embed_input);\r\n    if(resp && resp.usage) {\r\n      console.log(\"API key is valid\");\r\n      return true;\r\n    }else{\r\n      console.log(\"API key is invalid\");\r\n      return false;\r\n    }\r\n  }\r\n\r\n\r\n  output_render_log() {\r\n    // if settings.log_render is true\r\n    if(this.settings.log_render) {\r\n      if (this.render_log.new_embeddings === 0) {\r\n        return;\r\n      }else{\r\n        // pretty print this.render_log to console\r\n        console.log(JSON.stringify(this.render_log, null, 2));\r\n      }\r\n    }\r\n\r\n    // clear render_log\r\n    this.render_log = {};\r\n    this.render_log.deleted_embeddings = 0;\r\n    this.render_log.exclusions_logs = {};\r\n    this.render_log.failed_embeddings = [];\r\n    this.render_log.files = [];\r\n    this.render_log.new_embeddings = 0;\r\n    this.render_log.skipped_low_delta = {};\r\n    this.render_log.token_usage = 0;\r\n    this.render_log.tokens_saved_by_cache = 0;\r\n  }\r\n\r\n  // find connections by most similar to current note by cosine similarity\r\n  async find_note_connections(current_note=null) {\r\n    // md5 of current note path\r\n    const curr_key = md5(current_note.path);\r\n    // if in this.nearest_cache then set to nearest\r\n    // else get nearest\r\n    let nearest = [];\r\n    if(this.nearest_cache[curr_key]) {\r\n      nearest = this.nearest_cache[curr_key];\r\n      // console.log(\"nearest from cache\");\r\n    }else{\r\n      // skip files where path contains any exclusions\r\n      for(let j = 0; j < this.file_exclusions.length; j++) {\r\n        if(current_note.path.indexOf(this.file_exclusions[j]) > -1) {\r\n          this.log_exclusion(this.file_exclusions[j]);\r\n          // break out of loop and finish here\r\n          return \"excluded\";\r\n        }\r\n      }\r\n      // get all embeddings\r\n      // await this.get_all_embeddings();\r\n      // wrap get all in setTimeout to allow for UI to update\r\n      setTimeout(() => {\r\n        this.get_all_embeddings()\r\n      }, 3000);\r\n      // get from cache if mtime is same and values are not empty\r\n      if(this.smart_vec_lite.mtime_is_current(curr_key, current_note.stat.mtime)) {\r\n        // skipping get file embeddings because nothing has changed\r\n        // console.log(\"find_note_connections - skipping file (mtime)\");\r\n      }else{\r\n        // get file embeddings\r\n        await this.get_file_embeddings(current_note);\r\n      }\r\n      // get current note embedding vector\r\n      const vec = this.smart_vec_lite.get_vec(curr_key);\r\n      if(!vec) {\r\n        return \"Error getting embeddings for: \"+current_note.path;\r\n      }\r\n\r\n      // compute cosine similarity between current note and all other notes via embeddings\r\n      nearest = this.smart_vec_lite.nearest(vec, {\r\n        skip_key: curr_key,\r\n        skip_sections: this.settings.skip_sections,\r\n      });\r\n\r\n      // save to this.nearest_cache\r\n      this.nearest_cache[curr_key] = nearest;\r\n    }\r\n\r\n    // return array sorted by cosine similarity\r\n    return nearest;\r\n  }\r\n\r\n  // create render_log object of exlusions with number of times skipped as value\r\n  log_exclusion(exclusion) {\r\n    // increment render_log for skipped file\r\n    this.render_log.exclusions_logs[exclusion] = (this.render_log.exclusions_logs[exclusion] || 0) + 1;\r\n  }\r\n\r\n\r\n  block_parser(markdown, file_path){\r\n    // if this.settings.skip_sections is true then return empty array\r\n    if(this.settings.skip_sections) {\r\n      return [];\r\n    }\r\n    // split the markdown into lines\r\n    const lines = markdown.split('\\n');\r\n    // initialize the blocks array\r\n    let blocks = [];\r\n    // current headers array\r\n    let currentHeaders = [];\r\n    // remove .md file extension and convert file_path to breadcrumb formatting\r\n    const file_breadcrumbs = file_path.replace('.md', '').replace(/\\//g, ' > ');\r\n    // initialize the block string\r\n    let block = '';\r\n    let block_headings = '';\r\n    let block_path = file_path;\r\n\r\n    let last_heading_line = 0;\r\n    let i = 0;\r\n    let block_headings_list = [];\r\n    // loop through the lines\r\n    for (i = 0; i < lines.length; i++) {\r\n      // get the line\r\n      const line = lines[i];\r\n      // if line does not start with #\r\n      // or if line starts with # and second character is a word or number indicating a \"tag\"\r\n      // then add to block\r\n      if (!line.startsWith('#') || (['#',' '].indexOf(line[1]) < 0)){\r\n        // skip if line is empty\r\n        if(line === '') continue;\r\n        // skip if line is empty bullet or checkbox\r\n        if(['- ', '- [ ] '].indexOf(line) > -1) continue;\r\n        // if currentHeaders is empty skip (only blocks with headers, otherwise block.path conflicts with file.path)\r\n        if(currentHeaders.length === 0) continue;\r\n        // add line to block\r\n        block += \"\\n\" + line;\r\n        continue;\r\n      }\r\n      /**\r\n       * BEGIN Heading parsing\r\n       * - likely a heading if made it this far\r\n       */\r\n      last_heading_line = i;\r\n      // push the current block to the blocks array unless last line was a also a header\r\n      if(i > 0 && (last_heading_line !== (i-1)) && (block.indexOf(\"\\n\") > -1) && this.validate_headings(block_headings)) {\r\n        output_block();\r\n      }\r\n      // get the header level\r\n      const level = line.split('#').length - 1;\r\n      // remove any headers from the current headers array that are higher than the current header level\r\n      currentHeaders = currentHeaders.filter(header => header.level < level);\r\n      // add header and level to current headers array\r\n      // trim the header to remove \"#\" and any trailing spaces\r\n      currentHeaders.push({header: line.replace(/#/g, '').trim(), level: level});\r\n      // initialize the block breadcrumbs with file.path the current headers\r\n      block = file_breadcrumbs;\r\n      block += \": \" + currentHeaders.map(header => header.header).join(' > ');\r\n      block_headings = \"#\"+currentHeaders.map(header => header.header).join('#');\r\n      // if block_headings is already in block_headings_list then add a number to the end\r\n      if(block_headings_list.indexOf(block_headings) > -1) {\r\n        let count = 1;\r\n        while(block_headings_list.indexOf(`${block_headings}{${count}}`) > -1) {\r\n          count++;\r\n        }\r\n        block_headings = `${block_headings}{${count}}`;\r\n      }\r\n      block_headings_list.push(block_headings);\r\n      block_path = file_path + block_headings;\r\n    }\r\n    // handle remaining after loop\r\n    if((last_heading_line !== (i-1)) && (block.indexOf(\"\\n\") > -1) && this.validate_headings(block_headings)) output_block();\r\n    // remove any blocks that are too short (length < 50)\r\n    blocks = blocks.filter(b => b.length > 50);\r\n    // console.log(blocks);\r\n    // return the blocks array\r\n    return blocks;\r\n\r\n    function output_block() {\r\n      // breadcrumbs length (first line of block)\r\n      const breadcrumbs_length = block.indexOf(\"\\n\") + 1;\r\n      const block_length = block.length - breadcrumbs_length;\r\n      // trim block to max length\r\n      if (block.length > MAX_EMBED_STRING_LENGTH) {\r\n        block = block.substring(0, MAX_EMBED_STRING_LENGTH);\r\n      }\r\n      blocks.push({ text: block.trim(), path: block_path, length: block_length });\r\n    }\r\n  }\r\n  // reverse-retrieve block given path\r\n  async block_retriever(path, limits={}) {\r\n    limits = {\r\n      lines: null,\r\n      chars_per_line: null,\r\n      max_chars: null,\r\n      ...limits\r\n    }\r\n    // return if no # in path\r\n    if (path.indexOf('#') < 0) {\r\n      console.log(\"not a block path: \"+path);\r\n      return false;\r\n    }\r\n    let block = [];\r\n    let block_headings = path.split('#').slice(1);\r\n    // if path ends with number in curly braces\r\n    let heading_occurrence = 0;\r\n    if(block_headings[block_headings.length-1].indexOf('{') > -1) {\r\n      // get the occurrence number\r\n      heading_occurrence = parseInt(block_headings[block_headings.length-1].split('{')[1].replace('}', ''));\r\n      // remove the occurrence from the last heading\r\n      block_headings[block_headings.length-1] = block_headings[block_headings.length-1].split('{')[0];\r\n    }\r\n    let currentHeaders = [];\r\n    let occurrence_count = 0;\r\n    let begin_line = 0;\r\n    let i = 0;\r\n    // get file path from path\r\n    const file_path = path.split('#')[0];\r\n    // get file\r\n    const file = this.app.vault.getAbstractFileByPath(file_path);\r\n    if(!(file instanceof Obsidian.TFile)) {\r\n      console.log(\"not a file: \"+file_path);\r\n      return false;\r\n    }\r\n    // get file contents\r\n    const file_contents = await this.app.vault.cachedRead(file);\r\n    // split the file contents into lines\r\n    const lines = file_contents.split('\\n');\r\n    // loop through the lines\r\n    let is_code = false;\r\n    for (i = 0; i < lines.length; i++) {\r\n      // get the line\r\n      const line = lines[i];\r\n      // if line begins with three backticks then toggle is_code\r\n      if(line.indexOf('```') === 0) {\r\n        is_code = !is_code;\r\n      }\r\n      // if is_code is true then add line with preceding tab and continue\r\n      if(is_code) {\r\n        continue;\r\n      }\r\n      // skip if line is empty bullet or checkbox\r\n      if(['- ', '- [ ] '].indexOf(line) > -1) continue;\r\n      // if line does not start with #\r\n      // or if line starts with # and second character is a word or number indicating a \"tag\"\r\n      // then continue to next line\r\n      if (!line.startsWith('#') || (['#',' '].indexOf(line[1]) < 0)){\r\n        continue;\r\n      }\r\n      /**\r\n       * BEGIN Heading parsing\r\n       * - likely a heading if made it this far\r\n       */\r\n      // get the heading text\r\n      const heading_text = line.replace(/#/g, '').trim();\r\n      // continue if heading text is not in block_headings\r\n      const heading_index = block_headings.indexOf(heading_text);\r\n      if (heading_index < 0) continue;\r\n      // if currentHeaders.length !== heading_index then we have a mismatch\r\n      if (currentHeaders.length !== heading_index) continue;\r\n      // push the heading text to the currentHeaders array\r\n      currentHeaders.push(heading_text);\r\n      // if currentHeaders.length === block_headings.length then we have a match\r\n      if (currentHeaders.length === block_headings.length) {\r\n        // if heading_occurrence is defined then increment occurrence_count\r\n        if(heading_occurrence === 0) {\r\n          // set begin_line to i + 1\r\n          begin_line = i + 1;\r\n          break; // break out of loop\r\n        }\r\n        // if occurrence_count !== heading_occurrence then continue\r\n        if(occurrence_count === heading_occurrence){\r\n          begin_line = i + 1;\r\n          break; // break out of loop\r\n        }\r\n        occurrence_count++;\r\n        // reset currentHeaders\r\n        currentHeaders.pop();\r\n        continue;\r\n      }\r\n    }\r\n    // if no begin_line then return false\r\n    if (begin_line === 0) return false;\r\n    // iterate through lines starting at begin_line\r\n    is_code = false;\r\n    // character accumulator\r\n    let char_count = 0;\r\n    for (i = begin_line; i < lines.length; i++) {\r\n      if((typeof line_limit === \"number\") && (block.length > line_limit)){\r\n        block.push(\"...\");\r\n        break; // ends when line_limit is reached\r\n      }\r\n      let line = lines[i];\r\n      if ((line.indexOf('#') === 0) && (['#',' '].indexOf(line[1]) !== -1)){\r\n        break; // ends when encountering next header\r\n      }\r\n      // DEPRECATED: should be handled by new_line+char_count check (happens in previous iteration)\r\n      // if char_count is greater than limit.max_chars, skip\r\n      if (limits.max_chars && char_count > limits.max_chars) {\r\n        block.push(\"...\");\r\n        break;\r\n      }\r\n      // if new_line + char_count is greater than limit.max_chars, skip\r\n      if (limits.max_chars && ((line.length + char_count) > limits.max_chars)) {\r\n        const max_new_chars = limits.max_chars - char_count;\r\n        line = line.slice(0, max_new_chars) + \"...\";\r\n        break;\r\n      }\r\n      // validate/format\r\n      // if line is empty, skip\r\n      if (line.length === 0) continue;\r\n      // limit length of line to N characters\r\n      if (limits.chars_per_line && line.length > limits.chars_per_line) {\r\n        line = line.slice(0, limits.chars_per_line) + \"...\";\r\n      }\r\n      // if line is a code block, skip\r\n      if (line.startsWith(\"```\")) {\r\n        is_code = !is_code;\r\n        continue;\r\n      }\r\n      if (is_code){\r\n        // add tab to beginning of line\r\n        line = \"\\t\"+line;\r\n      }\r\n      // add line to block\r\n      block.push(line);\r\n      // increment char_count\r\n      char_count += line.length;\r\n    }\r\n    // close code block if open\r\n    if (is_code) {\r\n      block.push(\"```\");\r\n    }\r\n    return block.join(\"\\n\").trim();\r\n  }\r\n\r\n  // retrieve a file from the vault\r\n  async file_retriever(link, limits={}) {\r\n    limits = {\r\n      lines: null,\r\n      max_chars: null,\r\n      chars_per_line: null,\r\n      ...limits\r\n    };\r\n    const this_file = this.app.vault.getAbstractFileByPath(link);\r\n    // if file is not found, skip\r\n    if (!(this_file instanceof Obsidian.TAbstractFile)) return false;\r\n    // use cachedRead to get the first 10 lines of the file\r\n    const file_content = await this.app.vault.cachedRead(this_file);\r\n    const file_lines = file_content.split(\"\\n\");\r\n    let first_ten_lines = [];\r\n    let is_code = false;\r\n    let char_accum = 0;\r\n    const line_limit = limits.lines || file_lines.length;\r\n    for (let i = 0; first_ten_lines.length < line_limit; i++) {\r\n      let line = file_lines[i];\r\n      // if line is undefined, break\r\n      if (typeof line === 'undefined')\r\n        break;\r\n      // if line is empty, skip\r\n      if (line.length === 0)\r\n        continue;\r\n      // limit length of line to N characters\r\n      if (limits.chars_per_line && line.length > limits.chars_per_line) {\r\n        line = line.slice(0, limits.chars_per_line) + \"...\";\r\n      }\r\n      // if line is \"---\", skip\r\n      if (line === \"---\")\r\n        continue;\r\n      // skip if line is empty bullet or checkbox\r\n      if (['- ', '- [ ] '].indexOf(line) > -1)\r\n        continue;\r\n      // if line is a code block, skip\r\n      if (line.indexOf(\"```\") === 0) {\r\n        is_code = !is_code;\r\n        continue;\r\n      }\r\n      // if char_accum is greater than limit.max_chars, skip\r\n      if (limits.max_chars && char_accum > limits.max_chars) {\r\n        first_ten_lines.push(\"...\");\r\n        break;\r\n      }\r\n      if (is_code) {\r\n        // if is code, add tab to beginning of line\r\n        line = \"\\t\" + line;\r\n      }\r\n      // if line is a heading\r\n      if (line_is_heading(line)) {\r\n        // look at last line in first_ten_lines to see if it is a heading\r\n        // note: uses last in first_ten_lines, instead of look ahead in file_lines, because..\r\n        // ...next line may be excluded from first_ten_lines by previous if statements\r\n        if ((first_ten_lines.length > 0) && line_is_heading(first_ten_lines[first_ten_lines.length - 1])) {\r\n          // if last line is a heading, remove it\r\n          first_ten_lines.pop();\r\n        }\r\n      }\r\n      // add line to first_ten_lines\r\n      first_ten_lines.push(line);\r\n      // increment char_accum\r\n      char_accum += line.length;\r\n    }\r\n    // for each line in first_ten_lines, apply view-specific formatting\r\n    for (let i = 0; i < first_ten_lines.length; i++) {\r\n      // if line is a heading\r\n      if (line_is_heading(first_ten_lines[i])) {\r\n        // if this is the last line in first_ten_lines\r\n        if (i === first_ten_lines.length - 1) {\r\n          // remove the last line if it is a heading\r\n          first_ten_lines.pop();\r\n          break;\r\n        }\r\n        // remove heading syntax to improve readability in small space\r\n        first_ten_lines[i] = first_ten_lines[i].replace(/#+/, \"\");\r\n        first_ten_lines[i] = `\\n${first_ten_lines[i]}:`;\r\n      }\r\n    }\r\n    // join first ten lines into string\r\n    first_ten_lines = first_ten_lines.join(\"\\n\");\r\n    return first_ten_lines;\r\n  }\r\n\r\n  // iterate through blocks and skip if block_headings contains this.header_exclusions\r\n  validate_headings(block_headings) {\r\n    let valid = true;\r\n    if (this.header_exclusions.length > 0) {\r\n      for (let k = 0; k < this.header_exclusions.length; k++) {\r\n        if (block_headings.indexOf(this.header_exclusions[k]) > -1) {\r\n          valid = false;\r\n          this.log_exclusion(\"heading: \"+this.header_exclusions[k]);\r\n          break;\r\n        }\r\n      }\r\n    }\r\n    return valid;\r\n  }\r\n  // render \"Smart Connections\" text fixed in the bottom right corner\r\n  render_brand(container, location=\"default\") {\r\n    // if location is all then get Object.keys(this.sc_branding) and call this function for each\r\n    if (container === \"all\") {\r\n      const locations = Object.keys(this.sc_branding);\r\n      for (let i = 0; i < locations.length; i++) {\r\n        this.render_brand(this.sc_branding[locations[i]], locations[i]);\r\n      }\r\n      return;\r\n    }\r\n    // brand container\r\n    this.sc_branding[location] = container;\r\n    // if this.sc_branding[location] contains child with class \"sc-brand\", remove it\r\n    if (this.sc_branding[location].querySelector(\".sc-brand\")) {\r\n      this.sc_branding[location].querySelector(\".sc-brand\").remove();\r\n    }\r\n    const brand_container = this.sc_branding[location].createEl(\"div\", { cls: \"sc-brand\" });\r\n    // add text\r\n    // add SVG signal icon using getIcon\r\n    Obsidian.setIcon(brand_container, \"smart-connections\");\r\n    const brand_p = brand_container.createEl(\"p\");\r\n    let text = \"Smart Connections\";\r\n    let attr = {};\r\n    // if update available, change text to \"Update Available\"\r\n    if (this.update_available) {\r\n      text = \"Update Available\";\r\n      attr = {\r\n        style: \"font-weight: 700;\"\r\n      };\r\n    }\r\n    brand_p.createEl(\"a\", {\r\n      cls: \"\",\r\n      text: text,\r\n      href: \"https://github.com/brianpetro/obsidian-smart-connections/discussions\",\r\n      target: \"_blank\",\r\n      attr: attr\r\n    });\r\n  }\r\n\r\n\r\n  // create list of nearest notes\r\n  async update_results(container, nearest) {\r\n    let list;\r\n    // check if list exists\r\n    if((container.children.length > 1) && (container.children[1].classList.contains(\"sc-list\"))){\r\n      list = container.children[1];\r\n    }\r\n    // if list exists, empty it\r\n    if (list) {\r\n      list.empty();\r\n    } else {\r\n      // create list element\r\n      list = container.createEl(\"div\", { cls: \"sc-list\" });\r\n    }\r\n    let search_result_class = \"search-result\";\r\n    // if settings expanded_view is false, add sc-collapsed class\r\n    if(!this.settings.expanded_view) search_result_class += \" sc-collapsed\";\r\n\r\n    // TODO: add option to group nearest by file\r\n    if(!this.settings.group_nearest_by_file) {\r\n      // for each nearest note\r\n      for (let i = 0; i < nearest.length; i++) {\r\n        /**\r\n         * BEGIN EXTERNAL LINK LOGIC\r\n         * if link is an object, it indicates external link\r\n         */\r\n        console.log(this);\r\n        if (typeof nearest[i].link === \"object\") {\r\n          const item = list.createEl(\"div\", { cls: \"search-result\" });\r\n          const link = item.createEl(\"a\", {\r\n            cls: \"search-result-file-title is-clickable\",\r\n            href: nearest[i].link.path,\r\n            title: nearest[i].link.title,\r\n          });\r\n          link.innerHTML = this.render_external_link_elm(nearest[i].link);\r\n          item.setAttr('draggable', 'true')\r\n          continue; // ends here for external links\r\n        }\r\n        /**\r\n         * BEGIN INTERNAL LINK LOGIC\r\n         * if link is a string, it indicates internal link\r\n         */\r\n        let file_link_text;\r\n        const file_similarity_pct = Math.round(nearest[i].similarity * 100) + \"%\";\r\n        if(this.settings.show_full_path) {\r\n          const pcs = nearest[i].link.split(\"/\");\r\n          file_link_text = pcs[pcs.length - 1];\r\n          const path = pcs.slice(0, pcs.length - 1).join(\"/\");\r\n          // file_link_text = `<small>${path} | ${file_similarity_pct}</small><br>${file_link_text}`;\r\n          file_link_text = `<small>${file_similarity_pct} | ${path} | ${file_link_text}</small>`;\r\n        }else{\r\n          file_link_text = '<small>' + file_similarity_pct + \" | \" + nearest[i].link.split(\"/\").pop() + '</small>';\r\n        }\r\n        // skip contents rendering if incompatible file type\r\n        // ex. not markdown file or contains no '.excalidraw'\r\n        if(!this.renderable_file_type(nearest[i].link)){\r\n          const item = list.createEl(\"div\", { cls: \"search-result\" });\r\n          const link = item.createEl(\"a\", {\r\n            cls: \"search-result-file-title is-clickable\",\r\n            href: nearest[i].link,\r\n          });\r\n          link.innerHTML = file_link_text;\r\n          // drag and drop\r\n          item.setAttr('draggable', 'true')\r\n          // add listeners to link\r\n          this.add_link_listeners(link, nearest[i], item);\r\n          continue;\r\n        }\r\n\r\n        // remove file extension if .md and make # into >\r\n        file_link_text = file_link_text.replace(\".md\", \"\").replace(/#/g, \" > \");\r\n        // create item\r\n        const item = list.createEl(\"div\", { cls: search_result_class });\r\n        // create span for toggle\r\n        const toggle = item.createEl(\"span\", { cls: \"is-clickable\" });\r\n        // insert right triangle svg as toggle\r\n        Obsidian.setIcon(toggle, \"right-triangle\"); // must come before adding other elms to prevent overwrite\r\n        const link = toggle.createEl(\"a\", {\r\n          cls: \"search-result-file-title\",\r\n          title: nearest[i].link,\r\n        });\r\n        link.innerHTML = file_link_text;\r\n        // add listeners to link\r\n        this.add_link_listeners(link, nearest[i], item);\r\n        toggle.addEventListener(\"click\", (event) => {\r\n          // find parent containing search-result class\r\n          let parent = event.target.parentElement;\r\n          while (!parent.classList.contains(\"search-result\")) {\r\n            parent = parent.parentElement;\r\n          }\r\n          // toggle sc-collapsed class\r\n          parent.classList.toggle(\"sc-collapsed\");\r\n        });\r\n        const contents = item.createEl(\"ul\", { cls: \"\" });\r\n        const contents_container = contents.createEl(\"li\", {\r\n          cls: \"search-result-file-title is-clickable\",\r\n          title: nearest[i].link,\r\n        });\r\n        if(nearest[i].link.indexOf(\"#\") > -1){ // is block\r\n          Obsidian.MarkdownRenderer.renderMarkdown((await this.block_retriever(nearest[i].link, {lines: 10, max_chars: 1000})), contents_container, nearest[i].link, new Obsidian.Component());\r\n        }else{ // is file\r\n          const first_ten_lines = await this.file_retriever(nearest[i].link, {lines: 10, max_chars: 1000});\r\n          if(!first_ten_lines) continue; // skip if file is empty\r\n          Obsidian.MarkdownRenderer.renderMarkdown(first_ten_lines, contents_container, nearest[i].link, new Obsidian.Component());\r\n        }\r\n        this.add_link_listeners(contents, nearest[i], item);\r\n      }\r\n      this.render_brand(container, \"block\");\r\n      return;\r\n    }\r\n\r\n    // group nearest by file\r\n    const nearest_by_file = {};\r\n    for (let i = 0; i < nearest.length; i++) {\r\n      const curr = nearest[i];\r\n      const link = curr.link;\r\n      // skip if link is an object (indicates external logic)\r\n      if (typeof link === \"object\") {\r\n        nearest_by_file[link.path] = [curr];\r\n        continue;\r\n      }\r\n      if (link.indexOf(\"#\") > -1) {\r\n        const file_path = link.split(\"#\")[0];\r\n        if (!nearest_by_file[file_path]) {\r\n          nearest_by_file[file_path] = [];\r\n        }\r\n        nearest_by_file[file_path].push(nearest[i]);\r\n      } else {\r\n        if (!nearest_by_file[link]) {\r\n          nearest_by_file[link] = [];\r\n        }\r\n        // always add to front of array\r\n        nearest_by_file[link].unshift(nearest[i]);\r\n      }\r\n    }\r\n    // for each file\r\n    const keys = Object.keys(nearest_by_file);\r\n    for (let i = 0; i < keys.length; i++) {\r\n      const file = nearest_by_file[keys[i]];\r\n      /**\r\n       * Begin external link handling\r\n       */\r\n      // if link is an object (indicates v2 logic)\r\n      if (typeof file[0].link === \"object\") {\r\n        const curr = file[0];\r\n        const meta = curr.link;\r\n        if (meta.path.startsWith(\"http\")) {\r\n          const item = list.createEl(\"div\", { cls: \"search-result\" });\r\n          const link = item.createEl(\"a\", {\r\n            cls: \"search-result-file-title is-clickable\",\r\n            href: meta.path,\r\n            title: meta.title,\r\n          });\r\n          link.innerHTML = this.render_external_link_elm(meta);\r\n          item.setAttr('draggable', 'true');\r\n          continue; // ends here for external links\r\n        }\r\n      }\r\n      /**\r\n       * Handles Internal\r\n       */\r\n      let file_link_text;\r\n      const file_similarity_pct = Math.round(file[0].similarity * 100) + \"%\";\r\n      if (this.settings.show_full_path) {\r\n        const pcs = file[0].link.split(\"/\");\r\n        file_link_text = pcs[pcs.length - 1];\r\n        const path = pcs.slice(0, pcs.length - 1).join(\"/\");\r\n        file_link_text = `<small>${path} | ${file_similarity_pct}</small><br>${file_link_text}`;\r\n      } else {\r\n        file_link_text = file[0].link.split(\"/\").pop();\r\n        // add similarity percentage\r\n        file_link_text += ' | ' + file_similarity_pct;\r\n      }\r\n\r\n\r\n\r\n      // skip contents rendering if incompatible file type\r\n      // ex. not markdown or contains '.excalidraw'\r\n      if(!this.renderable_file_type(file[0].link)) {\r\n        const item = list.createEl(\"div\", { cls: \"search-result\" });\r\n        const file_link = item.createEl(\"a\", {\r\n          cls: \"search-result-file-title is-clickable\",\r\n          title: file[0].link,\r\n        });\r\n        file_link.innerHTML = file_link_text;\r\n        // add link listeners to file link\r\n        this.add_link_listeners(file_link, file[0], item);\r\n        continue;\r\n      }\r\n\r\n\r\n      // remove file extension if .md\r\n      file_link_text = file_link_text.replace(\".md\", \"\").replace(/#/g, \" > \");\r\n      const item = list.createEl(\"div\", { cls: search_result_class });\r\n      const toggle = item.createEl(\"span\", { cls: \"is-clickable\" });\r\n      // insert right triangle svg icon as toggle button in span\r\n      Obsidian.setIcon(toggle, \"right-triangle\"); // must come before adding other elms else overwrites\r\n      const file_link = toggle.createEl(\"a\", {\r\n        cls: \"search-result-file-title\",\r\n        title: file[0].link,\r\n      });\r\n      file_link.innerHTML = file_link_text;\r\n      // add link listeners to file link\r\n      this.add_link_listeners(file_link, file[0], toggle);\r\n      toggle.addEventListener(\"click\", (event) => {\r\n        // find parent containing class search-result\r\n        let parent = event.target;\r\n        while (!parent.classList.contains(\"search-result\")) {\r\n          parent = parent.parentElement;\r\n        }\r\n        parent.classList.toggle(\"sc-collapsed\");\r\n        // TODO: if block container is empty, render markdown from block retriever\r\n      });\r\n      const file_link_list = item.createEl(\"ul\");\r\n      // for each link in file\r\n      for (let j = 0; j < file.length; j++) {\r\n        // if is a block (has # in link)\r\n        if(file[j].link.indexOf(\"#\") > -1) {\r\n          const block = file[j];\r\n          const block_link = file_link_list.createEl(\"li\", {\r\n            cls: \"search-result-file-title is-clickable\",\r\n            title: block.link,\r\n          });\r\n          // skip block context if file.length === 1 because already added\r\n          if(file.length > 1) {\r\n            const block_context = this.render_block_context(block);\r\n            const block_similarity_pct = Math.round(block.similarity * 100) + \"%\";\r\n            block_link.innerHTML = `<small>${block_context} | ${block_similarity_pct}</small>`;\r\n          }\r\n          const block_container = block_link.createEl(\"div\");\r\n          // TODO: move to rendering on expanding section (toggle collapsed)\r\n          Obsidian.MarkdownRenderer.renderMarkdown((await this.block_retriever(block.link, {lines: 10, max_chars: 1000})), block_container, block.link, new Obsidian.Component());\r\n          // add link listeners to block link\r\n          this.add_link_listeners(block_link, block, file_link_list);\r\n        }else{\r\n          // get first ten lines of file\r\n          const file_link_list = item.createEl(\"ul\");\r\n          const block_link = file_link_list.createEl(\"li\", {\r\n            cls: \"search-result-file-title is-clickable\",\r\n            title: file[0].link,\r\n          });\r\n          const block_container = block_link.createEl(\"div\");\r\n          let first_ten_lines = await this.file_retriever(file[0].link, {lines: 10, max_chars: 1000});\r\n          if(!first_ten_lines) continue; // if file not found, skip\r\n          Obsidian.MarkdownRenderer.renderMarkdown(first_ten_lines, block_container, file[0].link, new Obsidian.Component());\r\n          this.add_link_listeners(block_link, file[0], file_link_list);\r\n\r\n        }\r\n      }\r\n    }\r\n    this.render_brand(container, \"file\");\r\n  }\r\n\r\n  add_link_listeners(item, curr, list) {\r\n    item.addEventListener(\"click\", async (event) => {\r\n      await this.open_note(curr, event);\r\n    });\r\n    // drag-on\r\n    // currently only works with full-file links\r\n    item.setAttr('draggable', 'true');\r\n    item.addEventListener('dragstart', (event) => {\r\n      const dragManager = this.app.dragManager;\r\n      const file_path = curr.link.split(\"#\")[0];\r\n      const file = this.app.metadataCache.getFirstLinkpathDest(file_path, '');\r\n      const dragData = dragManager.dragFile(event, file);\r\n      // console.log(dragData);\r\n      dragManager.onDragStart(event, dragData);\r\n    });\r\n    // if curr.link contains curly braces, return (incompatible with hover-link)\r\n    if (curr.link.indexOf(\"{\") > -1) return;\r\n    // trigger hover event on link\r\n    item.addEventListener(\"mouseover\", (event) => {\r\n      this.app.workspace.trigger(\"hover-link\", {\r\n        event,\r\n        source: SMART_CONNECTIONS_VIEW_TYPE,\r\n        hoverParent: list,\r\n        targetEl: item,\r\n        linktext: curr.link,\r\n      });\r\n    });\r\n  }\r\n\r\n  // get target file from link path\r\n  // if sub-section is linked, open file and scroll to sub-section\r\n  async open_note(curr, event=null) {\r\n    let targetFile;\r\n    let heading;\r\n    if (curr.link.indexOf(\"#\") > -1) {\r\n      // remove after # from link\r\n      targetFile = this.app.metadataCache.getFirstLinkpathDest(curr.link.split(\"#\")[0], \"\");\r\n      // console.log(targetFile);\r\n      const target_file_cache = this.app.metadataCache.getFileCache(targetFile);\r\n      // console.log(target_file_cache);\r\n      // get heading\r\n      let heading_text = curr.link.split(\"#\").pop();\r\n      // if heading text contains a curly brace, get the number inside the curly braces as occurence\r\n      let occurence = 0;\r\n      if (heading_text.indexOf(\"{\") > -1) {\r\n        // get occurence\r\n        occurence = parseInt(heading_text.split(\"{\")[1].split(\"}\")[0]);\r\n        // remove occurence from heading text\r\n        heading_text = heading_text.split(\"{\")[0];\r\n      }\r\n      // get headings from file cache\r\n      const headings = target_file_cache.headings;\r\n      // get headings with the same depth and text as the link\r\n      for(let i = 0; i < headings.length; i++) {\r\n        if (headings[i].heading === heading_text) {\r\n          // if occurence is 0, set heading and break\r\n          if(occurence === 0) {\r\n            heading = headings[i];\r\n            break;\r\n          }\r\n          occurence--; // decrement occurence\r\n        }\r\n      }\r\n      // console.log(heading);\r\n    } else {\r\n      targetFile = this.app.metadataCache.getFirstLinkpathDest(curr.link, \"\");\r\n    }\r\n    let leaf;\r\n    if(event) {\r\n      // properly handle if the meta/ctrl key is pressed\r\n      const mod = Obsidian.Keymap.isModEvent(event);\r\n      // get most recent leaf\r\n      leaf = this.app.workspace.getLeaf(mod);\r\n    }else{\r\n      // get most recent leaf\r\n      leaf = this.app.workspace.getMostRecentLeaf();\r\n    }\r\n    await leaf.openFile(targetFile);\r\n    if (heading) {\r\n      let { editor } = leaf.view;\r\n      const pos = { line: heading.position.start.line, ch: 0 };\r\n      editor.setCursor(pos);\r\n      editor.scrollIntoView({ to: pos, from: pos }, true);\r\n    }\r\n  }\r\n\r\n  render_block_context(block) {\r\n    const block_headings = block.link.split(\".md\")[1].split(\"#\");\r\n    // starting with the last heading first, iterate through headings\r\n    let block_context = \"\";\r\n    for (let i = block_headings.length - 1; i >= 0; i--) {\r\n      if(block_context.length > 0) {\r\n        block_context = ` > ${block_context}`;\r\n      }\r\n      block_context = block_headings[i] + block_context;\r\n      // if block context is longer than N characters, break\r\n      if (block_context.length > 100) {\r\n        break;\r\n      }\r\n    }\r\n    // remove leading > if exists\r\n    if (block_context.startsWith(\" > \")) {\r\n      block_context = block_context.slice(3);\r\n    }\r\n    return block_context;\r\n\r\n  }\r\n\r\n  renderable_file_type(link) {\r\n    return (link.indexOf(\".md\") !== -1) && (link.indexOf(\".excalidraw\") === -1);\r\n  }\r\n\r\n  render_external_link_elm(meta){\r\n    if(meta.source) {\r\n      if(meta.source === \"Gmail\") meta.source = \"\uD83D\uDCE7 Gmail\";\r\n      return `<small>${meta.source}</small><br>${meta.title}`;\r\n    }\r\n    // remove http(s)://\r\n    let domain = meta.path.replace(/(^\\w+:|^)\\/\\//, \"\");\r\n    // separate domain from path\r\n    domain = domain.split(\"/\")[0];\r\n    // wrap domain in <small> and add line break\r\n    return `<small>\uD83C\uDF10 ${domain}</small><br>${meta.title}`;\r\n  }\r\n  // get all folders\r\n  async get_all_folders() {\r\n    if(!this.folders || this.folders.length === 0){\r\n      this.folders = await this.get_folders();\r\n    }\r\n    return this.folders;\r\n  }\r\n  // get folders, traverse non-hidden sub-folders\r\n  async get_folders(path = \"/\") {\r\n    let folders = (await this.app.vault.adapter.list(path)).folders;\r\n    let folder_list = [];\r\n    for (let i = 0; i < folders.length; i++) {\r\n      if (folders[i].startsWith(\".\")) continue;\r\n      folder_list.push(folders[i]);\r\n      folder_list = folder_list.concat(await this.get_folders(folders[i] + \"/\"));\r\n    }\r\n    return folder_list;\r\n  }\r\n\r\n\r\n  async sync_notes() {\r\n    // if license key is not set, return\r\n    if(!this.settings.license_key){\r\n      new Obsidian.Notice(\"Smart Connections: Supporter license key is required to sync notes to the ChatGPT Plugin server.\");\r\n      return;\r\n    }\r\n    console.log(\"syncing notes\");\r\n    // get all files in vault\r\n    const files = this.app.vault.getMarkdownFiles().filter((file) => {\r\n      // filter out file paths matching any strings in this.file_exclusions\r\n      for(let i = 0; i < this.file_exclusions.length; i++) {\r\n        if(file.path.indexOf(this.file_exclusions[i]) > -1) {\r\n          return false;\r\n        }\r\n      }\r\n      return true;\r\n    });\r\n    const notes = await this.build_notes_object(files);\r\n    console.log(\"object built\");\r\n    // save notes object to .smart-connections/notes.json\r\n    await this.app.vault.adapter.write(\".smart-connections/notes.json\", JSON.stringify(notes, null, 2));\r\n    console.log(\"notes saved\");\r\n    console.log(this.settings.license_key);\r\n    // POST notes object to server\r\n    const response = await (0, Obsidian.requestUrl)({\r\n      url: \"https://sync.smartconnections.app/sync\",\r\n      method: \"POST\",\r\n      headers: {\r\n        \"Content-Type\": \"application/json\",\r\n      },\r\n      contentType: \"application/json\",\r\n      body: JSON.stringify({\r\n        license_key: this.settings.license_key,\r\n        notes: notes\r\n      })\r\n    });\r\n    console.log(response);\r\n\r\n  }\r\n\r\n  async build_notes_object(files) {\r\n    let output = {};\r\n\r\n    for(let i = 0; i < files.length; i++) {\r\n      let file = files[i];\r\n      let parts = file.path.split(\"/\");\r\n      let current = output;\r\n\r\n      for (let ii = 0; ii < parts.length; ii++) {\r\n        let part = parts[ii];\r\n\r\n        if (ii === parts.length - 1) {\r\n          // This is a file\r\n          current[part] = await this.app.vault.cachedRead(file);\r\n        } else {\r\n          // This is a directory\r\n          if (!current[part]) {\r\n            current[part] = {};\r\n          }\r\n\r\n          current = current[part];\r\n        }\r\n      }\r\n    }\r\n\r\n    return output;\r\n  }\r\n\r\n}\r\n\r\nconst SMART_CONNECTIONS_VIEW_TYPE = \"smart-connections-view\";\r\nclass SmartConnectionsView extends Obsidian.ItemView {\r\n  constructor(leaf, plugin) {\r\n    super(leaf);\r\n    this.plugin = plugin;\r\n    this.nearest = null;\r\n    this.load_wait = null;\r\n  }\r\n  getViewType() {\r\n    return SMART_CONNECTIONS_VIEW_TYPE;\r\n  }\r\n\r\n  getDisplayText() {\r\n    return \"Smart Connections Files\";\r\n  }\r\n\r\n  getIcon() {\r\n    return \"smart-connections\";\r\n  }\r\n\r\n\r\n  set_message(message) {\r\n    const container = this.containerEl.children[1];\r\n    // clear container\r\n    container.empty();\r\n    // initiate top bar\r\n    this.initiate_top_bar(container);\r\n    // if mesage is an array, loop through and create a new p element for each message\r\n    if (Array.isArray(message)) {\r\n      for (let i = 0; i < message.length; i++) {\r\n        container.createEl(\"p\", { cls: \"sc_message\", text: message[i] });\r\n      }\r\n    }else{\r\n      // create p element with message\r\n      container.createEl(\"p\", { cls: \"sc_message\", text: message });\r\n    }\r\n  }\r\n  render_link_text(link, show_full_path=false) {\r\n    /**\r\n     * Begin internal links\r\n     */\r\n    // if show full path is false, remove file path\r\n    if (!show_full_path) {\r\n      link = link.split(\"/\").pop();\r\n    }\r\n    // if contains '#'\r\n    if (link.indexOf(\"#\") > -1) {\r\n      // split at .md\r\n      link = link.split(\".md\");\r\n      // wrap first part in <small> and add line break\r\n      link[0] = `<small>${link[0]}</small><br>`;\r\n      // join back together\r\n      link = link.join(\"\");\r\n      // replace '#' with ' \u00BB '\r\n      link = link.replace(/\\#/g, \" \u00BB \");\r\n    }else{\r\n      // remove '.md'\r\n      link = link.replace(\".md\", \"\");\r\n    }\r\n    return link;\r\n  }\r\n\r\n\r\n  set_nearest(nearest, nearest_context=null, results_only=false) {\r\n    // get container element\r\n    const container = this.containerEl.children[1];\r\n    // if results only is false, clear container and initiate top bar\r\n    if(!results_only){\r\n      // clear container\r\n      container.empty();\r\n      this.initiate_top_bar(container, nearest_context);\r\n    }\r\n    // update results\r\n    this.plugin.update_results(container, nearest);\r\n  }\r\n\r\n  initiate_top_bar(container, nearest_context=null) {\r\n    let top_bar;\r\n    // if top bar already exists, empty it\r\n    if ((container.children.length > 0) && (container.children[0].classList.contains(\"sc-top-bar\"))) {\r\n      top_bar = container.children[0];\r\n      top_bar.empty();\r\n    } else {\r\n      // init container for top bar\r\n      top_bar = container.createEl(\"div\", { cls: \"sc-top-bar\" });\r\n    }\r\n    // if highlighted text is not null, create p element with highlighted text\r\n    if (nearest_context) {\r\n      top_bar.createEl(\"p\", { cls: \"sc-context\", text: nearest_context });\r\n    }\r\n    // add chat button\r\n    const chat_button = top_bar.createEl(\"button\", { cls: \"sc-chat-button\" });\r\n    // add icon to chat button\r\n    Obsidian.setIcon(chat_button, \"message-square\");\r\n    // add click listener to chat button\r\n    chat_button.addEventListener(\"click\", () => {\r\n      // open chat\r\n      this.plugin.open_chat();\r\n    });\r\n    // add search button\r\n    const search_button = top_bar.createEl(\"button\", { cls: \"sc-search-button\" });\r\n    // add icon to search button\r\n    Obsidian.setIcon(search_button, \"search\");\r\n    // add click listener to search button\r\n    search_button.addEventListener(\"click\", () => {\r\n      // empty top bar\r\n      top_bar.empty();\r\n      // create input element\r\n      const search_container = top_bar.createEl(\"div\", { cls: \"search-input-container\" });\r\n      const input = search_container.createEl(\"input\", {\r\n        cls: \"sc-search-input\",\r\n        type: \"search\",\r\n        placeholder: \"Type to start search...\",\r\n      });\r\n      // focus input\r\n      input.focus();\r\n      // add keydown listener to input\r\n      input.addEventListener(\"keydown\", (event) => {\r\n        // if escape key is pressed\r\n        if (event.key === \"Escape\") {\r\n          this.clear_auto_searcher();\r\n          // clear top bar\r\n          this.initiate_top_bar(container, nearest_context);\r\n        }\r\n      });\r\n\r\n      // add keyup listener to input\r\n      input.addEventListener(\"keyup\", (event) => {\r\n        // if this.search_timeout is not null then clear it and set to null\r\n        this.clear_auto_searcher();\r\n        // get search term\r\n        const search_term = input.value;\r\n        // if enter key is pressed\r\n        if (event.key === \"Enter\" && search_term !== \"\") {\r\n          this.search(search_term);\r\n        }\r\n        // if any other key is pressed and input is not empty then wait 500ms and make_connections\r\n        else if (search_term !== \"\") {\r\n          // clear timeout\r\n          clearTimeout(this.search_timeout);\r\n          // set timeout\r\n          this.search_timeout = setTimeout(() => {\r\n            this.search(search_term, true);\r\n          }, 700);\r\n        }\r\n      });\r\n    });\r\n  }\r\n\r\n  // render buttons: \"create\" and \"retry\" for loading embeddings.json file\r\n  render_embeddings_buttons() {\r\n    // get container element\r\n    const container = this.containerEl.children[1];\r\n    // clear container\r\n    container.empty();\r\n    // create heading that says \"Embeddings file not found\"\r\n    container.createEl(\"h2\", { cls: \"scHeading\", text: \"Embeddings file not found\" });\r\n    // create div for buttons\r\n    const button_div = container.createEl(\"div\", { cls: \"scButtonDiv\" });\r\n    // create \"create\" button\r\n    const create_button = button_div.createEl(\"button\", { cls: \"scButton\", text: \"Create embeddings.json\" });\r\n    // note that creating embeddings.json file will trigger bulk embedding and may take a while\r\n    button_div.createEl(\"p\", { cls: \"scButtonNote\", text: \"Warning: Creating embeddings.json file will trigger bulk embedding and may take a while\" });\r\n    // create \"retry\" button\r\n    const retry_button = button_div.createEl(\"button\", { cls: \"scButton\", text: \"Retry\" });\r\n    // try to load embeddings.json file again\r\n    button_div.createEl(\"p\", { cls: \"scButtonNote\", text: \"If embeddings.json file already exists, click 'Retry' to load it\" });\r\n\r\n    // add click event to \"create\" button\r\n    create_button.addEventListener(\"click\", async (event) => {\r\n      // create embeddings.json file\r\n      await this.plugin.smart_vec_lite.init_embeddings_file();\r\n      // reload view\r\n      await this.render_connections();\r\n    });\r\n\r\n    // add click event to \"retry\" button\r\n    retry_button.addEventListener(\"click\", async (event) => {\r\n      console.log(\"retrying to load embeddings.json file\");\r\n      // reload embeddings.json file\r\n      await this.plugin.init_vecs();\r\n      // reload view\r\n      await this.render_connections();\r\n    });\r\n  }\r\n\r\n  async onOpen() {\r\n    const container = this.containerEl.children[1];\r\n    container.empty();\r\n    // placeholder text\r\n    container.createEl(\"p\", { cls: \"scPlaceholder\", text: \"Open a note to find connections.\" });\r\n\r\n    // runs when file is opened\r\n    this.plugin.registerEvent(this.app.workspace.on('file-open', (file) => {\r\n      // if no file is open, return\r\n      if(!file) {\r\n        // console.log(\"no file open, returning\");\r\n        return;\r\n      }\r\n      // return if file type is not supported\r\n      if(SUPPORTED_FILE_TYPES.indexOf(file.extension) === -1) {\r\n        return this.set_message([\r\n          \"File: \"+file.name\r\n          ,\"Unsupported file type (Supported: \"+SUPPORTED_FILE_TYPES.join(\", \")+\")\"\r\n        ]);\r\n      }\r\n      // run render_connections after 1 second to allow for file to load\r\n      if(this.load_wait){\r\n        clearTimeout(this.load_wait);\r\n      }\r\n      this.load_wait = setTimeout(() => {\r\n        this.render_connections(file);\r\n        this.load_wait = null;\r\n      }, 1000);\r\n\r\n    }));\r\n\r\n    this.app.workspace.registerHoverLinkSource(SMART_CONNECTIONS_VIEW_TYPE, {\r\n        display: 'Smart Connections Files',\r\n        defaultMod: true,\r\n    });\r\n    this.app.workspace.registerHoverLinkSource(SMART_CONNECTIONS_CHAT_VIEW_TYPE, {\r\n        display: 'Smart Chat Links',\r\n        defaultMod: true,\r\n    });\r\n\r\n    this.app.workspace.onLayoutReady(this.initialize.bind(this));\r\n\r\n  }\r\n\r\n  async initialize() {\r\n    this.set_message(\"Loading embeddings file...\");\r\n    const vecs_intiated = await this.plugin.init_vecs();\r\n    if(vecs_intiated){\r\n      this.set_message(\"Embeddings file loaded.\");\r\n      await this.render_connections();\r\n    }else{\r\n      this.render_embeddings_buttons();\r\n    }\r\n\r\n    /**\r\n     * EXPERIMENTAL\r\n     * - window-based API access\r\n     * - code-block rendering\r\n     */\r\n    this.api = new SmartConnectionsViewApi(this.app, this.plugin, this);\r\n    // register API to global window object\r\n    (window[\"SmartConnectionsViewApi\"] = this.api) && this.register(() => delete window[\"SmartConnectionsViewApi\"]);\r\n\r\n  }\r\n\r\n  async onClose() {\r\n    console.log(\"closing smart connections view\");\r\n    this.app.workspace.unregisterHoverLinkSource(SMART_CONNECTIONS_VIEW_TYPE);\r\n    this.plugin.view = null;\r\n  }\r\n\r\n  async render_connections(context=null) {\r\n    console.log(\"rendering connections\");\r\n    // if API key is not set then update view message\r\n    if(!this.plugin.settings.api_key) {\r\n      this.set_message(\"An OpenAI API key is required to make Smart Connections\");\r\n      return;\r\n    }\r\n    if(!this.plugin.embeddings_loaded){\r\n      await this.plugin.init_vecs();\r\n    }\r\n    // if embedding still not loaded, return\r\n    if(!this.plugin.embeddings_loaded) {\r\n      console.log(\"embeddings files still not loaded or yet to be created\");\r\n      this.render_embeddings_buttons();\r\n      return;\r\n    }\r\n    this.set_message(\"Making Smart Connections...\");\r\n    /**\r\n     * Begin highlighted-text-level search\r\n     */\r\n    if(typeof context === \"string\") {\r\n      const highlighted_text = context;\r\n      // get embedding for highlighted text\r\n      await this.search(highlighted_text);\r\n      return; // ends here if context is a string\r\n    }\r\n\r\n    /**\r\n     * Begin file-level search\r\n     */\r\n    this.nearest = null;\r\n    this.interval_count = 0;\r\n    this.rendering = false;\r\n    this.file = context;\r\n    // if this.interval is set then clear it\r\n    if(this.interval) {\r\n      clearInterval(this.interval);\r\n      this.interval = null;\r\n    }\r\n    // set interval to check if nearest is set\r\n    this.interval = setInterval(() => {\r\n      if(!this.rendering){\r\n        if(this.file instanceof Obsidian.TFile) {\r\n          this.rendering = true;\r\n          this.render_note_connections(this.file);\r\n        }else{\r\n          // get current note\r\n          this.file = this.app.workspace.getActiveFile();\r\n          // if still no current note then return\r\n          if(!this.file && this.count > 1) {\r\n            clearInterval(this.interval);\r\n            this.set_message(\"No active file\");\r\n            return;\r\n          }\r\n        }\r\n      }else{\r\n        if(this.nearest) {\r\n          clearInterval(this.interval);\r\n          // if nearest is a string then update view message\r\n          if (typeof this.nearest === \"string\") {\r\n            this.set_message(this.nearest);\r\n          } else {\r\n            // set nearest connections\r\n            this.set_nearest(this.nearest, \"File: \" + this.file.name);\r\n          }\r\n          // if render_log.failed_embeddings then update failed_embeddings.txt\r\n          if (this.plugin.render_log.failed_embeddings.length > 0) {\r\n            this.plugin.save_failed_embeddings();\r\n          }\r\n          // get object keys of render_log\r\n          this.plugin.output_render_log();\r\n          return;\r\n        }else{\r\n          this.interval_count++;\r\n          this.set_message(\"Making Smart Connections...\"+this.interval_count);\r\n        }\r\n      }\r\n    }, 10);\r\n  }\r\n\r\n  async render_note_connections(file) {\r\n    this.nearest = await this.plugin.find_note_connections(file);\r\n  }\r\n\r\n  clear_auto_searcher() {\r\n    if (this.search_timeout) {\r\n      clearTimeout(this.search_timeout);\r\n      this.search_timeout = null;\r\n    }\r\n  }\r\n\r\n  async search(search_text, results_only=false) {\r\n    const nearest = await this.plugin.api.search(search_text);\r\n    // render results in view with first 100 characters of search text\r\n    const nearest_context = `Selection: \"${search_text.length > 100 ? search_text.substring(0, 100) + \"...\" : search_text}\"`;\r\n    this.set_nearest(nearest, nearest_context, results_only);\r\n  }\r\n\r\n}\r\nclass SmartConnectionsViewApi {\r\n  constructor(app, plugin, view) {\r\n    this.app = app;\r\n    this.plugin = plugin;\r\n    this.view = view;\r\n  }\r\n  async search(search_text) {\r\n    return await this.plugin.api.search(search_text);\r\n  }\r\n  // trigger reload of embeddings file\r\n  async reload_embeddings_file() {\r\n    await this.plugin.init_vecs();\r\n    await this.view.render_connections();\r\n  }\r\n  async init_vecs() {\r\n    this.smart_vec_lite = new VecLite({\r\n      folder_path: \".smart-connections\",\r\n      exists_adapter: this.app.vault.adapter.exists.bind(\r\n        this.app.vault.adapter\r\n      ),\r\n      mkdir_adapter: this.app.vault.adapter.mkdir.bind(this.app.vault.adapter),\r\n      read_adapter: this.app.vault.adapter.read.bind(this.app.vault.adapter),\r\n      rename_adapter: this.app.vault.adapter.rename.bind(\r\n        this.app.vault.adapter\r\n      ),\r\n      stat_adapter: this.app.vault.adapter.stat.bind(this.app.vault.adapter),\r\n      write_adapter: this.app.vault.adapter.write.bind(this.app.vault.adapter),\r\n    });\r\n    this.embeddings_loaded = await this.smart_vec_lite.load();\r\n    return this.embeddings_loaded;\r\n  }\r\n}\r\nclass ScSearchApi {\r\n  constructor(app, plugin) {\r\n    this.app = app;\r\n    this.plugin = plugin;\r\n  }\r\n  async search (search_text, filter={}) {\r\n    filter = {\r\n      skip_sections: this.plugin.settings.skip_sections,\r\n      ...filter\r\n    }\r\n    let nearest = [];\r\n    const resp = await this.plugin.request_embedding_from_input(search_text);\r\n    if (resp && resp.data && resp.data[0] && resp.data[0].embedding) {\r\n      nearest = this.plugin.smart_vec_lite.nearest(resp.data[0].embedding, filter);\r\n    } else {\r\n      // resp is null, undefined, or missing data\r\n      new Obsidian.Notice(\"Smart Connections: Error getting embedding\");\r\n    }\r\n    return nearest;\r\n  }\r\n}\r\n\r\nclass SmartConnectionsSettingsTab extends Obsidian.PluginSettingTab {\r\n  constructor(app, plugin) {\r\n    super(app, plugin);\r\n    this.plugin = plugin;\r\n  }\r\n  display() {\r\n    const {\r\n      containerEl\r\n    } = this;\r\n    containerEl.empty();\r\n    containerEl.createEl(\"h2\", {\r\n      text: \"Supporter Settings\"\r\n    });\r\n    // list supporter benefits\r\n    containerEl.createEl(\"p\", {\r\n      text: \"As a Smart Connections \\\"Supporter\\\", fast-track your PKM journey with priority perks and pioneering innovations.\"\r\n    });\r\n    // three list items\r\n    const supporter_benefits_list = containerEl.createEl(\"ul\");\r\n    supporter_benefits_list.createEl(\"li\", {\r\n      text: \"Enjoy swift, top-priority support.\"\r\n    });\r\n    supporter_benefits_list.createEl(\"li\", {\r\n      text: \"Gain early access to version 2 (includes local embedding model).\"\r\n    });\r\n    supporter_benefits_list.createEl(\"li\", {\r\n      text: \"Stay informed and engaged with exclusive supporter-only communications.\"\r\n    });\r\n    // add a text input to enter supporter license key\r\n    new Obsidian.Setting(containerEl).setName(\"Supporter License Key\").setDesc(\"Note: this is not required to use Smart Connections.\").addText((text) => text.setPlaceholder(\"Enter your license_key\").setValue(this.plugin.settings.license_key).onChange(async (value) => {\r\n      this.plugin.settings.license_key = value.trim();\r\n      await this.plugin.saveSettings(true);\r\n    }));\r\n    // button \"get v2\"\r\n    new Obsidian.Setting(containerEl).setName(\"Get v2\").setDesc(\"Get v2 (warning: very early beta release, likely to crash, please send issues directly to the supporter email for quick response)\").addButton((button) => button.setButtonText(\"Get v2 (unstable)\").onClick(async () => {\r\n      await this.plugin.update_to_v2();\r\n    }));\r\n    // add button to trigger sync notes to use with ChatGPT\r\n    new Obsidian.Setting(containerEl).setName(\"Sync Notes\").setDesc(\"Make notes available via the Smart Connections ChatGPT Plugin. Respects exclusion settings configured below.\").addButton((button) => button.setButtonText(\"Sync Notes\").onClick(async () => {\r\n      // sync notes\r\n      await this.plugin.sync_notes();\r\n    }));\r\n    // add button to become a supporter\r\n    new Obsidian.Setting(containerEl).setName(\"Become a Supporter\").setDesc(\"Become a Supporter\").addButton((button) => button.setButtonText(\"Become a Supporter\").onClick(async () => {\r\n      const payment_pages = [\r\n        \"https://buy.stripe.com/9AQ5kO5QnbAWgGAbIY\",\r\n        \"https://buy.stripe.com/9AQ7sWemT48u1LGcN4\"\r\n      ];\r\n      if(!this.plugin.payment_page_index){\r\n        this.plugin.payment_page_index = Math.round(Math.random());\r\n      }\r\n      // open supporter page in browser\r\n      window.open(payment_pages[this.plugin.payment_page_index]);\r\n    }));\r\n\r\n\r\n    containerEl.createEl(\"h2\", {\r\n      text: \"OpenAI Settings\"\r\n    });\r\n    // add a text input to enter the API key\r\n    new Obsidian.Setting(containerEl).setName(\"OpenAI API Key\").setDesc(\"Required: an OpenAI API key is currently required to use Smart Connections.\").addText((text) => text.setPlaceholder(\"Enter your api_key\").setValue(this.plugin.settings.api_key).onChange(async (value) => {\r\n      this.plugin.settings.api_key = value.trim();\r\n      await this.plugin.saveSettings(true);\r\n    }));\r\n    // add a button to test the API key is working\r\n    new Obsidian.Setting(containerEl).setName(\"Test API Key\").setDesc(\"Test API Key\").addButton((button) => button.setButtonText(\"Test API Key\").onClick(async () => {\r\n      // test API key\r\n      const resp = await this.plugin.test_api_key();\r\n      if(resp) {\r\n        new Obsidian.Notice(\"Smart Connections: API key is valid\");\r\n      }else{\r\n        new Obsidian.Notice(\"Smart Connections: API key is not working as expected!\");\r\n      }\r\n    }));\r\n    // add dropdown to select the model\r\n    new Obsidian.Setting(containerEl).setName(\"Smart Chat Model\").setDesc(\"Select a model to use with Smart Chat.\").addDropdown((dropdown) => {\r\n      dropdown.addOption(\"gpt-3.5-turbo-16k\", \"gpt-3.5-turbo-16k\");\r\n      dropdown.addOption(\"gpt-4\", \"gpt-4 (limited access, 8k)\");\r\n      dropdown.addOption(\"gpt-3.5-turbo\", \"gpt-3.5-turbo (4k)\");\r\n      dropdown.addOption(\"gpt-4-1106-preview\", \"gpt-4-turbo (128k)\");\r\n      dropdown.onChange(async (value) => {\r\n        this.plugin.settings.smart_chat_model = value;\r\n        await this.plugin.saveSettings();\r\n      });\r\n      dropdown.setValue(this.plugin.settings.smart_chat_model);\r\n    });\r\n    // language\r\n    new Obsidian.Setting(containerEl).setName(\"Default Language\").setDesc(\"Default language to use for Smart Chat. Changes which self-referential pronouns will trigger lookup of your notes.\").addDropdown((dropdown) => {\r\n      // get Object keys from pronous\r\n      const languages = Object.keys(SMART_TRANSLATION);\r\n      for(let i = 0; i < languages.length; i++) {\r\n        dropdown.addOption(languages[i], languages[i]);\r\n      }\r\n      dropdown.onChange(async (value) => {\r\n        this.plugin.settings.language = value;\r\n        await this.plugin.saveSettings();\r\n        self_ref_pronouns_list.setText(this.get_self_ref_list());\r\n        // if chat view is open then run new_chat()\r\n        const chat_view = this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE).length > 0 ? this.app.workspace.getLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE)[0].view : null;\r\n        if(chat_view) {\r\n          chat_view.new_chat();\r\n        }\r\n      });\r\n      dropdown.setValue(this.plugin.settings.language);\r\n    });\r\n    // list current self-referential pronouns\r\n    const self_ref_pronouns_list = containerEl.createEl(\"span\", {\r\n      text: this.get_self_ref_list()\r\n    });\r\n    new Obsidian.Setting(containerEl).setName(\"Cut off frontmatter\").setDesc(\"Cut off frontmatter in the prompt to gain characters in reply generation\").addToggle((toggle) => { toggle.setValue(this.plugin.settings.cut_off_frontmatter).onChange(async (value) => { this.plugin.settings.cut_off_frontmatter = value; await this.plugin.saveSettings(); }); });\r\n    containerEl.createEl(\"h2\", {\r\n      text: \"Exclusions\"\r\n    });\r\n    // list file exclusions\r\n    new Obsidian.Setting(containerEl).setName(\"file_exclusions\").setDesc(\"'Excluded file' matchers separated by a comma.\").addText((text) => text.setPlaceholder(\"drawings,prompts/logs\").setValue(this.plugin.settings.file_exclusions).onChange(async (value) => {\r\n      this.plugin.settings.file_exclusions = value;\r\n      await this.plugin.saveSettings();\r\n    }));\r\n    // list folder exclusions\r\n    new Obsidian.Setting(containerEl).setName(\"folder_exclusions\").setDesc(\"'Excluded folder' matchers separated by a comma.\").addText((text) => text.setPlaceholder(\"drawings,prompts/logs\").setValue(this.plugin.settings.folder_exclusions).onChange(async (value) => {\r\n      this.plugin.settings.folder_exclusions = value;\r\n      await this.plugin.saveSettings();\r\n    }));\r\n    // list path only matchers\r\n    new Obsidian.Setting(containerEl).setName(\"path_only\").setDesc(\"'Path only' matchers separated by a comma.\").addText((text) => text.setPlaceholder(\"drawings,prompts/logs\").setValue(this.plugin.settings.path_only).onChange(async (value) => {\r\n      this.plugin.settings.path_only = value;\r\n      await this.plugin.saveSettings();\r\n    }));\r\n    // list header exclusions\r\n    new Obsidian.Setting(containerEl).setName(\"header_exclusions\").setDesc(\"'Excluded header' matchers separated by a comma. Works for 'blocks' only.\").addText((text) => text.setPlaceholder(\"drawings,prompts/logs\").setValue(this.plugin.settings.header_exclusions).onChange(async (value) => {\r\n      this.plugin.settings.header_exclusions = value;\r\n      await this.plugin.saveSettings();\r\n    }));\r\n    containerEl.createEl(\"h2\", {\r\n      text: \"Display\"\r\n    });\r\n    // toggle showing full path in view\r\n    new Obsidian.Setting(containerEl).setName(\"show_full_path\").setDesc(\"Show full path in view.\").addToggle((toggle) => toggle.setValue(this.plugin.settings.show_full_path).onChange(async (value) => {\r\n      this.plugin.settings.show_full_path = value;\r\n      await this.plugin.saveSettings(true);\r\n    }));\r\n    // toggle expanded view by default\r\n    new Obsidian.Setting(containerEl).setName(\"expanded_view\").setDesc(\"Expanded view by default.\").addToggle((toggle) => toggle.setValue(this.plugin.settings.expanded_view).onChange(async (value) => {\r\n      this.plugin.settings.expanded_view = value;\r\n      await this.plugin.saveSettings(true);\r\n    }));\r\n    // toggle group nearest by file\r\n    new Obsidian.Setting(containerEl).setName(\"group_nearest_by_file\").setDesc(\"Group nearest by file.\").addToggle((toggle) => toggle.setValue(this.plugin.settings.group_nearest_by_file).onChange(async (value) => {\r\n      this.plugin.settings.group_nearest_by_file = value;\r\n      await this.plugin.saveSettings(true);\r\n    }));\r\n    // toggle view_open on Obsidian startup\r\n    new Obsidian.Setting(containerEl).setName(\"view_open\").setDesc(\"Open view on Obsidian startup.\").addToggle((toggle) => toggle.setValue(this.plugin.settings.view_open).onChange(async (value) => {\r\n      this.plugin.settings.view_open = value;\r\n      await this.plugin.saveSettings(true);\r\n    }));\r\n    // toggle chat_open on Obsidian startup\r\n    new Obsidian.Setting(containerEl).setName(\"chat_open\").setDesc(\"Open view on Obsidian startup.\").addToggle((toggle) => toggle.setValue(this.plugin.settings.chat_open).onChange(async (value) => {\r\n      this.plugin.settings.chat_open = value;\r\n      await this.plugin.saveSettings(true);\r\n    }));\r\n    // open in big view or small view\r\n    new Obsidian.Setting(containerEl).setName(\"open_in_big_view\").setDesc(\"Open in big view or small view.\").addDropdown((dropdown) => {\r\n      dropdown.addOption(false, \"Right pane (small)\");\r\n      dropdown.addOption(true, \"Main pane (big)\");\r\n      dropdown.setValue(this.plugin.settings.open_in_big_view);\r\n      dropdown.onChange(async (value) => {\r\n        this.plugin.settings.open_in_big_view = JSON.parse(value);\r\n        await this.plugin.saveSettings(true);\r\n        this.plugin.open_chat();\r\n\r\n      });\r\n    });\r\n    containerEl.createEl(\"h2\", {\r\n      text: \"Advanced\"\r\n    });\r\n    // toggle log_render\r\n    new Obsidian.Setting(containerEl).setName(\"log_render\").setDesc(\"Log render details to console (includes token_usage).\").addToggle((toggle) => toggle.setValue(this.plugin.settings.log_render).onChange(async (value) => {\r\n      this.plugin.settings.log_render = value;\r\n      await this.plugin.saveSettings(true);\r\n    }));\r\n    // toggle files in log_render\r\n    new Obsidian.Setting(containerEl).setName(\"log_render_files\").setDesc(\"Log embedded objects paths with log render (for debugging).\").addToggle((toggle) => toggle.setValue(this.plugin.settings.log_render_files).onChange(async (value) => {\r\n      this.plugin.settings.log_render_files = value;\r\n      await this.plugin.saveSettings(true);\r\n    }));\r\n    // toggle skip_sections\r\n    new Obsidian.Setting(containerEl).setName(\"skip_sections\").setDesc(\"Skips making connections to specific sections within notes. Warning: reduces usefulness for large files and requires 'Force Refresh' for sections to work in the future.\").addToggle((toggle) => toggle.setValue(this.plugin.settings.skip_sections).onChange(async (value) => {\r\n      this.plugin.settings.skip_sections = value;\r\n      await this.plugin.saveSettings(true);\r\n    }));\r\n    // test file writing by creating a test file, then writing additional data to the file, and returning any error text if it fails\r\n    containerEl.createEl(\"h3\", {\r\n      text: \"Test File Writing\"\r\n    });\r\n    // manual save button\r\n    containerEl.createEl(\"h3\", {\r\n      text: \"Manual Save\"\r\n    });\r\n    let manual_save_results = containerEl.createEl(\"div\");\r\n    new Obsidian.Setting(containerEl).setName(\"manual_save\").setDesc(\"Save current embeddings\").addButton((button) => button.setButtonText(\"Manual Save\").onClick(async () => {\r\n      // confirm\r\n      if (confirm(\"Are you sure you want to save your current embeddings?\")) {\r\n        // save\r\n        try{\r\n          await this.plugin.save_embeddings_to_file(true);\r\n          manual_save_results.innerHTML = \"Embeddings saved successfully.\";\r\n        }catch(e){\r\n          manual_save_results.innerHTML = \"Embeddings failed to save. Error: \" + e;\r\n        }\r\n      }\r\n    }));\r\n\r\n    // list previously failed files\r\n    containerEl.createEl(\"h3\", {\r\n      text: \"Previously failed files\"\r\n    });\r\n    let failed_list = containerEl.createEl(\"div\");\r\n    this.draw_failed_files_list(failed_list);\r\n\r\n    // force refresh button\r\n    containerEl.createEl(\"h3\", {\r\n      text: \"Force Refresh\"\r\n    });\r\n    new Obsidian.Setting(containerEl).setName(\"force_refresh\").setDesc(\"WARNING: DO NOT use unless you know what you are doing! This will delete all of your current embeddings from OpenAI and trigger reprocessing of your entire vault!\").addButton((button) => button.setButtonText(\"Force Refresh\").onClick(async () => {\r\n      // confirm\r\n      if (confirm(\"Are you sure you want to Force Refresh? By clicking yes you confirm that you understand the consequences of this action.\")) {\r\n        // force refresh\r\n        await this.plugin.force_refresh_embeddings_file();\r\n      }\r\n    }));\r\n\r\n  }\r\n  get_self_ref_list() {\r\n    return \"Current: \" + SMART_TRANSLATION[this.plugin.settings.language].pronous.join(\", \");\r\n  }\r\n\r\n  draw_failed_files_list(failed_list) {\r\n    failed_list.empty();\r\n    if(this.plugin.settings.failed_files.length > 0) {\r\n      // add message that these files will be skipped until manually retried\r\n      failed_list.createEl(\"p\", {\r\n        text: \"The following files failed to process and will be skipped until manually retried.\"\r\n      });\r\n      let list = failed_list.createEl(\"ul\");\r\n      for (let failed_file of this.plugin.settings.failed_files) {\r\n        list.createEl(\"li\", {\r\n          text: failed_file\r\n        });\r\n      }\r\n      // add button to retry failed files only\r\n      new Obsidian.Setting(failed_list).setName(\"retry_failed_files\").setDesc(\"Retry failed files only\").addButton((button) => button.setButtonText(\"Retry failed files only\").onClick(async () => {\r\n        // clear failed_list element\r\n        failed_list.empty();\r\n        // set \"retrying\" text\r\n        failed_list.createEl(\"p\", {\r\n          text: \"Retrying failed files...\"\r\n        });\r\n        await this.plugin.retry_failed_files();\r\n        // redraw failed files list\r\n        this.draw_failed_files_list(failed_list);\r\n      }));\r\n    }else{\r\n      failed_list.createEl(\"p\", {\r\n        text: \"No failed files\"\r\n      });\r\n    }\r\n  }\r\n}\r\n\r\nfunction line_is_heading(line) {\r\n  return (line.indexOf(\"#\") === 0) && (['#', ' '].indexOf(line[1]) !== -1);\r\n}\r\n\r\nconst SMART_CONNECTIONS_CHAT_VIEW_TYPE = \"smart-connections-chat-view\";\r\n\r\nclass SmartConnectionsChatView extends Obsidian.ItemView {\r\n  constructor(leaf, plugin) {\r\n    super(leaf);\r\n    this.plugin = plugin;\r\n    this.active_elm = null;\r\n    this.active_stream = null;\r\n    this.brackets_ct = 0;\r\n    this.chat = null;\r\n    this.chat_box = null;\r\n    this.chat_container = null;\r\n    this.current_chat_ml = [];\r\n    this.files = [];\r\n    this.last_from = null;\r\n    this.message_container = null;\r\n    this.prevent_input = false;\r\n  }\r\n  getDisplayText() {\r\n    return \"Smart Connections Chat\";\r\n  }\r\n  getIcon() {\r\n    return \"message-square\";\r\n  }\r\n  getViewType() {\r\n    return SMART_CONNECTIONS_CHAT_VIEW_TYPE;\r\n  }\r\n  onOpen() {\r\n    this.new_chat();\r\n    this.plugin.get_all_folders(); // sets this.plugin.folders necessary for folder-context\r\n  }\r\n  onClose() {\r\n    this.chat.save_chat();\r\n    this.app.workspace.unregisterHoverLinkSource(SMART_CONNECTIONS_CHAT_VIEW_TYPE);\r\n  }\r\n  render_chat() {\r\n    this.containerEl.empty();\r\n    this.chat_container = this.containerEl.createDiv(\"sc-chat-container\");\r\n    // render plus sign for clear button\r\n    this.render_top_bar();\r\n    // render chat messages container\r\n    this.render_chat_box();\r\n    // render chat input\r\n    this.render_chat_input();\r\n    this.plugin.render_brand(this.containerEl, \"chat\");\r\n  }\r\n  // render plus sign for clear button\r\n  render_top_bar() {\r\n    // create container for clear button\r\n    let top_bar_container = this.chat_container.createDiv(\"sc-top-bar-container\");\r\n    // render the name of the chat in an input box (pop content after last hyphen in chat_id)\r\n    let chat_name =this.chat.name();\r\n    let chat_name_input = top_bar_container.createEl(\"input\", {\r\n      attr: {\r\n        type: \"text\",\r\n        value: chat_name\r\n      },\r\n      cls: \"sc-chat-name-input\"\r\n    });\r\n    chat_name_input.addEventListener(\"change\", this.rename_chat.bind(this));\r\n\r\n    // create button to Smart View\r\n    let smart_view_btn = this.create_top_bar_button(top_bar_container, \"Smart View\", \"smart-connections\");\r\n    smart_view_btn.addEventListener(\"click\", this.open_smart_view.bind(this));\r\n    // create button to save chat\r\n    let save_btn = this.create_top_bar_button(top_bar_container, \"Save Chat\", \"save\");\r\n    save_btn.addEventListener(\"click\", this.save_chat.bind(this));\r\n    // create button to open chat history modal\r\n    let history_btn = this.create_top_bar_button(top_bar_container, \"Chat History\", \"history\");\r\n    history_btn.addEventListener(\"click\", this.open_chat_history.bind(this));\r\n    // create button to start new chat\r\n    const new_chat_btn = this.create_top_bar_button(top_bar_container, \"New Chat\", \"plus\");\r\n    new_chat_btn.addEventListener(\"click\", this.new_chat.bind(this));\r\n  }\r\n  async open_chat_history() {\r\n    const folder = await this.app.vault.adapter.list(\".smart-connections/chats\");\r\n    this.files = folder.files.map((file) => {\r\n      return file.replace(\".smart-connections/chats/\", \"\").replace(\".json\", \"\");\r\n    });\r\n    // open chat history modal\r\n    if (!this.modal)\r\n      this.modal = new SmartConnectionsChatHistoryModal(this.app, this);\r\n    this.modal.open();\r\n  }\r\n\r\n  create_top_bar_button(top_bar_container, title, icon=null) {\r\n    let btn = top_bar_container.createEl(\"button\", {\r\n      attr: {\r\n        title: title\r\n      }\r\n    });\r\n    if(icon){\r\n      Obsidian.setIcon(btn, icon);\r\n    }else{\r\n      btn.innerHTML = title;\r\n    }\r\n    return btn;\r\n  }\r\n  // render new chat\r\n  new_chat() {\r\n    this.clear_chat();\r\n    this.render_chat();\r\n    // render initial message from assistant (don't use render_message to skip adding to chat history)\r\n    this.new_messsage_bubble(\"assistant\");\r\n    this.active_elm.innerHTML = '<p>' + SMART_TRANSLATION[this.plugin.settings.language].initial_message+'</p>';\r\n  }\r\n  // open a chat from the chat history modal\r\n  async open_chat(chat_id) {\r\n    this.clear_chat();\r\n    await this.chat.load_chat(chat_id);\r\n    this.render_chat();\r\n    for (let i = 0; i < this.chat.chat_ml.length; i++) {\r\n      await this.render_message(this.chat.chat_ml[i].content, this.chat.chat_ml[i].role);\r\n    }\r\n  }\r\n  // clear current chat state\r\n  clear_chat() {\r\n    if (this.chat) {\r\n      this.chat.save_chat();\r\n    }\r\n    this.chat = new SmartConnectionsChatModel(this.plugin);\r\n    // if this.dotdotdot_interval is not null, clear interval\r\n    if (this.dotdotdot_interval) {\r\n      clearInterval(this.dotdotdot_interval);\r\n    }\r\n    // clear current chat ml\r\n    this.current_chat_ml = [];\r\n    // update prevent input\r\n    this.end_stream();\r\n  }\r\n\r\n  rename_chat(event) {\r\n    let new_chat_name = event.target.value;\r\n    this.chat.rename_chat(new_chat_name);\r\n  }\r\n\r\n  // save current chat\r\n  save_chat() {\r\n    this.chat.save_chat();\r\n    new Obsidian.Notice(\"[Smart Connections] Chat saved\");\r\n  }\r\n\r\n  open_smart_view() {\r\n    this.plugin.open_view();\r\n  }\r\n  // render chat messages container\r\n  render_chat_box() {\r\n    // create container for chat messages\r\n    this.chat_box = this.chat_container.createDiv(\"sc-chat-box\");\r\n    // create container for message\r\n    this.message_container = this.chat_box.createDiv(\"sc-message-container\");\r\n  }\r\n  // open file suggestion modal\r\n  open_file_suggestion_modal() {\r\n    // open file suggestion modal\r\n    if(!this.file_selector) this.file_selector = new SmartConnectionsFileSelectModal(this.app, this);\r\n    this.file_selector.open();\r\n  }\r\n  // open folder suggestion modal\r\n  async open_folder_suggestion_modal() {\r\n    // open folder suggestion modal\r\n    if(!this.folder_selector){\r\n      this.folder_selector = new SmartConnectionsFolderSelectModal(this.app, this);\r\n    }\r\n    this.folder_selector.open();\r\n  }\r\n  // insert_selection from file suggestion modal\r\n  insert_selection(insert_text) {\r\n    // get caret position\r\n    let caret_pos = this.textarea.selectionStart;\r\n    // get text before caret\r\n    let text_before = this.textarea.value.substring(0, caret_pos);\r\n    // get text after caret\r\n    let text_after = this.textarea.value.substring(caret_pos, this.textarea.value.length);\r\n    // insert text\r\n    this.textarea.value = text_before + insert_text + text_after;\r\n    // set caret position\r\n    this.textarea.selectionStart = caret_pos + insert_text.length;\r\n    this.textarea.selectionEnd = caret_pos + insert_text.length;\r\n    // focus on textarea\r\n    this.textarea.focus();\r\n  }\r\n\r\n  // render chat textarea and button\r\n  render_chat_input() {\r\n    // create container for chat input\r\n    let chat_input = this.chat_container.createDiv(\"sc-chat-form\");\r\n    // create textarea\r\n    this.textarea = chat_input.createEl(\"textarea\", {\r\n      cls: \"sc-chat-input\",\r\n      attr: {\r\n        placeholder: SMART_TRANSLATION[this.plugin.settings.language].try_placeholder\r\n      }\r\n    });\r\n    // use contenteditable instead of textarea\r\n    // this.textarea = chat_input.createEl(\"div\", {cls: \"sc-chat-input\", attr: {contenteditable: true}});\r\n    // add event listener to listen for shift+enter\r\n    chat_input.addEventListener(\"keyup\", (e) => {\r\n      if([\"[\", \"/\"].indexOf(e.key) === -1) return; // skip if key is not [ or /\r\n      const caret_pos = this.textarea.selectionStart;\r\n      // if key is open square bracket\r\n      if (e.key === \"[\") {\r\n        // if previous char is [\r\n        if(this.textarea.value[caret_pos - 2] === \"[\"){\r\n          // open file suggestion modal\r\n          this.open_file_suggestion_modal();\r\n          return;\r\n        }\r\n      }else{\r\n        this.brackets_ct = 0;\r\n      }\r\n      // if / is pressed\r\n      if (e.key === \"/\") {\r\n        // get caret position\r\n        // if this is first char or previous char is space\r\n        if (this.textarea.value.length === 1 || this.textarea.value[caret_pos - 2] === \" \") {\r\n          // open folder suggestion modal\r\n          this.open_folder_suggestion_modal();\r\n          return;\r\n        }\r\n      }\r\n\r\n    });\r\n\r\n    chat_input.addEventListener(\"keydown\", (e) => {\r\n      if (e.key === \"Enter\" && e.shiftKey) {\r\n        e.preventDefault();\r\n        if(this.prevent_input){\r\n          console.log(\"wait until current response is finished\");\r\n          new Obsidian.Notice(\"[Smart Connections] Wait until current response is finished\");\r\n          return;\r\n        }\r\n        // get text from textarea\r\n        let user_input = this.textarea.value;\r\n        // clear textarea\r\n        this.textarea.value = \"\";\r\n        // initiate response from assistant\r\n        this.initialize_response(user_input);\r\n      }\r\n      this.textarea.style.height = 'auto';\r\n      this.textarea.style.height = (this.textarea.scrollHeight) + 'px';\r\n    });\r\n    // button container\r\n    let button_container = chat_input.createDiv(\"sc-button-container\");\r\n    // create hidden abort button\r\n    let abort_button = button_container.createEl(\"span\", { attr: {id: \"sc-abort-button\", style: \"display: none;\"} });\r\n    Obsidian.setIcon(abort_button, \"square\");\r\n    // add event listener to button\r\n    abort_button.addEventListener(\"click\", () => {\r\n      // abort current response\r\n      this.end_stream();\r\n    });\r\n    // create button\r\n    let button = button_container.createEl(\"button\", { attr: {id: \"sc-send-button\"}, cls: \"send-button\" });\r\n    button.innerHTML = \"Send\";\r\n    // add event listener to button\r\n    button.addEventListener(\"click\", () => {\r\n      if(this.prevent_input){\r\n        console.log(\"wait until current response is finished\");\r\n        new Obsidian.Notice(\"Wait until current response is finished\");\r\n        return;\r\n      }\r\n      // get text from textarea\r\n      let user_input = this.textarea.value;\r\n      // clear textarea\r\n      this.textarea.value = \"\";\r\n      // initiate response from assistant\r\n      this.initialize_response(user_input);\r\n    });\r\n  }\r\n  async initialize_response(user_input) {\r\n    this.set_streaming_ux();\r\n    // render message\r\n    await this.render_message(user_input, \"user\");\r\n    this.chat.new_message_in_thread({\r\n      role: \"user\",\r\n      content: user_input\r\n    });\r\n    await this.render_dotdotdot();\r\n\r\n    // if contains internal link represented by [[link]]\r\n    if(this.chat.contains_internal_link(user_input)) {\r\n      this.chat.get_response_with_note_context(user_input, this);\r\n      return;\r\n    }\r\n    // // for testing purposes\r\n    // if(this.chat.contains_folder_reference(user_input)) {\r\n    //   const folders = this.chat.get_folder_references(user_input);\r\n    //   console.log(folders);\r\n    //   return;\r\n    // }\r\n    // if contains self referential keywords or folder reference\r\n    if(this.contains_self_referential_keywords(user_input) || this.chat.contains_folder_reference(user_input)) {\r\n      // get hyde\r\n      const context = await this.get_context_hyde(user_input);\r\n      // get user input with added context\r\n      // const context_input = this.build_context_input(context);\r\n      // console.log(context_input);\r\n      const chatml = [\r\n        {\r\n          role: \"system\",\r\n          // content: context_input\r\n          content: context\r\n        },\r\n        {\r\n          role: \"user\",\r\n          content: user_input\r\n        }\r\n      ];\r\n      this.request_chatgpt_completion({messages: chatml, temperature: 0});\r\n      return;\r\n    }\r\n    // completion without any specific context\r\n    this.request_chatgpt_completion();\r\n  }\r\n\r\n  async render_dotdotdot() {\r\n    if (this.dotdotdot_interval)\r\n      clearInterval(this.dotdotdot_interval);\r\n    await this.render_message(\"...\", \"assistant\");\r\n    // if is '...', then initiate interval to change to '.' and then to '..' and then to '...'\r\n    let dots = 0;\r\n    this.active_elm.innerHTML = '...';\r\n    this.dotdotdot_interval = setInterval(() => {\r\n      dots++;\r\n      if (dots > 3)\r\n        dots = 1;\r\n      this.active_elm.innerHTML = '.'.repeat(dots);\r\n    }, 500);\r\n    // wait 2 seconds for testing\r\n    // await new Promise(r => setTimeout(r, 2000));\r\n  }\r\n\r\n  set_streaming_ux() {\r\n    this.prevent_input = true;\r\n    // hide send button\r\n    if(document.getElementById(\"sc-send-button\"))\r\n      document.getElementById(\"sc-send-button\").style.display = \"none\";\r\n    // show abort button\r\n    if(document.getElementById(\"sc-abort-button\"))\r\n      document.getElementById(\"sc-abort-button\").style.display = \"block\";\r\n  }\r\n  unset_streaming_ux() {\r\n    this.prevent_input = false;\r\n    // show send button, remove display none\r\n    if(document.getElementById(\"sc-send-button\"))\r\n      document.getElementById(\"sc-send-button\").style.display = \"\";\r\n    // hide abort button\r\n    if(document.getElementById(\"sc-abort-button\"))\r\n      document.getElementById(\"sc-abort-button\").style.display = \"none\";\r\n  }\r\n\r\n\r\n  // check if includes keywords referring to one's own notes\r\n  contains_self_referential_keywords(user_input) {\r\n    const matches = user_input.match(this.plugin.self_ref_kw_regex);\r\n    if(matches) return true;\r\n    return false;\r\n  }\r\n\r\n  // render message\r\n  async render_message(message, from=\"assistant\", append_last=false) {\r\n    // if dotdotdot interval is set, then clear it\r\n    if(this.dotdotdot_interval) {\r\n      clearInterval(this.dotdotdot_interval);\r\n      this.dotdotdot_interval = null;\r\n      // clear last message\r\n      this.active_elm.innerHTML = '';\r\n    }\r\n    if(append_last) {\r\n      this.current_message_raw += message;\r\n      if(message.indexOf('\\n') === -1) {\r\n        this.active_elm.innerHTML += message;\r\n      }else{\r\n        this.active_elm.innerHTML = '';\r\n        // append to last message\r\n        await Obsidian.MarkdownRenderer.renderMarkdown(this.current_message_raw, this.active_elm, '?no-dataview', new Obsidian.Component());\r\n      }\r\n    }else{\r\n      this.current_message_raw = '';\r\n      if((this.chat.thread.length === 0) || (this.last_from !== from)) {\r\n        // create message\r\n        this.new_messsage_bubble(from);\r\n      }\r\n      // set message text\r\n      this.active_elm.innerHTML = '';\r\n      await Obsidian.MarkdownRenderer.renderMarkdown(message, this.active_elm, '?no-dataview', new Obsidian.Component());\r\n      // get links\r\n      this.handle_links_in_message();\r\n      // render button(s)\r\n      this.render_message_action_buttons(message);\r\n    }\r\n    // scroll to bottom\r\n    this.message_container.scrollTop = this.message_container.scrollHeight;\r\n  }\r\n  render_message_action_buttons(message) {\r\n    if (this.chat.context && this.chat.hyd) {\r\n      // render button to copy hyd in smart-connections code block\r\n      const context_view = this.active_elm.createEl(\"span\", {\r\n        cls: \"sc-msg-button\",\r\n        attr: {\r\n          title: \"Copy context to clipboard\" /* tooltip */\r\n        }\r\n      });\r\n      const this_hyd = this.chat.hyd;\r\n      Obsidian.setIcon(context_view, \"eye\");\r\n      context_view.addEventListener(\"click\", () => {\r\n        // copy to clipboard\r\n        navigator.clipboard.writeText(\"```smart-connections\\n\" + this_hyd + \"\\n```\\n\");\r\n        new Obsidian.Notice(\"[Smart Connections] Context code block copied to clipboard\");\r\n      });\r\n    }\r\n    if(this.chat.context) {\r\n      // render copy context button\r\n      const copy_prompt_button = this.active_elm.createEl(\"span\", {\r\n        cls: \"sc-msg-button\",\r\n        attr: {\r\n          title: \"Copy prompt to clipboard\" /* tooltip */\r\n        }\r\n      });\r\n      const this_context = this.chat.context.replace(/\\`\\`\\`/g, \"\\t```\").trimLeft();\r\n      Obsidian.setIcon(copy_prompt_button, \"files\");\r\n      copy_prompt_button.addEventListener(\"click\", () => {\r\n        // copy to clipboard\r\n        navigator.clipboard.writeText(\"```prompt-context\\n\" + this_context + \"\\n```\\n\");\r\n        new Obsidian.Notice(\"[Smart Connections] Context copied to clipboard\");\r\n      });\r\n    }\r\n    // render copy button\r\n    const copy_button = this.active_elm.createEl(\"span\", {\r\n      cls: \"sc-msg-button\",\r\n      attr: {\r\n        title: \"Copy message to clipboard\" /* tooltip */\r\n      }\r\n    });\r\n    Obsidian.setIcon(copy_button, \"copy\");\r\n    copy_button.addEventListener(\"click\", () => {\r\n      // copy message to clipboard\r\n      navigator.clipboard.writeText(message.trimLeft());\r\n      new Obsidian.Notice(\"[Smart Connections] Message copied to clipboard\");\r\n    });\r\n  }\r\n\r\n  handle_links_in_message() {\r\n    const links = this.active_elm.querySelectorAll(\"a\");\r\n    // if this active element contains a link\r\n    if (links.length > 0) {\r\n      for (let i = 0; i < links.length; i++) {\r\n        const link = links[i];\r\n        const link_text = link.getAttribute(\"data-href\");\r\n        // trigger hover event on link\r\n        link.addEventListener(\"mouseover\", (event) => {\r\n          this.app.workspace.trigger(\"hover-link\", {\r\n            event,\r\n            source: SMART_CONNECTIONS_CHAT_VIEW_TYPE,\r\n            hoverParent: link.parentElement,\r\n            targetEl: link,\r\n            // extract link text from a.data-href\r\n            linktext: link_text\r\n          });\r\n        });\r\n        // trigger open link event on link\r\n        link.addEventListener(\"click\", (event) => {\r\n          const link_tfile = this.app.metadataCache.getFirstLinkpathDest(link_text, \"/\");\r\n          // properly handle if the meta/ctrl key is pressed\r\n          const mod = Obsidian.Keymap.isModEvent(event);\r\n          // get most recent leaf\r\n          let leaf = this.app.workspace.getLeaf(mod);\r\n          leaf.openFile(link_tfile);\r\n        });\r\n      }\r\n    }\r\n  }\r\n\r\n  new_messsage_bubble(from) {\r\n    let message_el = this.message_container.createDiv(`sc-message ${from}`);\r\n    // create message content\r\n    this.active_elm = message_el.createDiv(\"sc-message-content\");\r\n    // set last from\r\n    this.last_from = from;\r\n  }\r\n\r\n  async request_chatgpt_completion(opts={}) {\r\n    const chat_ml = opts.messages || opts.chat_ml || this.chat.prepare_chat_ml();\r\n    console.log(\"chat_ml\", chat_ml);\r\n    const max_total_tokens = Math.round(get_max_chars(this.plugin.settings.smart_chat_model) / 4);\r\n    console.log(\"max_total_tokens\", max_total_tokens);\r\n    const curr_token_est = Math.round(JSON.stringify(chat_ml).length / 3);\r\n    console.log(\"curr_token_est\", curr_token_est);\r\n    let max_available_tokens = max_total_tokens - curr_token_est;\r\n    // if max_available_tokens is less than 0, set to 200\r\n    if(max_available_tokens < 0) max_available_tokens = 200;\r\n    else if(max_available_tokens > 4096) max_available_tokens = 4096;\r\n    console.log(\"max_available_tokens\", max_available_tokens);\r\n    opts = {\r\n      model: this.plugin.settings.smart_chat_model,\r\n      messages: chat_ml,\r\n      // max_tokens: 250,\r\n      max_tokens: max_available_tokens,\r\n      temperature: 0.3,\r\n      top_p: 1,\r\n      presence_penalty: 0,\r\n      frequency_penalty: 0,\r\n      stream: true,\r\n      stop: null,\r\n      n: 1,\r\n      // logit_bias: logit_bias,\r\n      ...opts\r\n    }\r\n    // console.log(opts.messages);\r\n    if(opts.stream) {\r\n      const full_str = await new Promise((resolve, reject) => {\r\n        try {\r\n          // console.log(\"stream\", opts);\r\n          const url = \"https://api.openai.com/v1/chat/completions\";\r\n          this.active_stream = new ScStreamer(url, {\r\n            headers: {\r\n              \"Content-Type\": \"application/json\",\r\n              Authorization: `Bearer ${this.plugin.settings.api_key}`\r\n            },\r\n            method: \"POST\",\r\n            payload: JSON.stringify(opts)\r\n          });\r\n          let txt = \"\";\r\n          this.active_stream.addEventListener(\"message\", (e) => {\r\n            if (e.data != \"[DONE]\") {\r\n              const payload = JSON.parse(e.data);\r\n              const text = payload.choices[0].delta.content;\r\n              if (!text) {\r\n                return;\r\n              }\r\n              txt += text;\r\n              this.render_message(text, \"assistant\", true);\r\n            } else {\r\n              this.end_stream();\r\n              resolve(txt);\r\n            }\r\n          });\r\n          this.active_stream.addEventListener(\"readystatechange\", (e) => {\r\n            if (e.readyState >= 2) {\r\n              console.log(\"ReadyState: \" + e.readyState);\r\n            }\r\n          });\r\n          this.active_stream.addEventListener(\"error\", (e) => {\r\n            console.error(e);\r\n            new Obsidian.Notice(\"Smart Connections Error Streaming Response. See console for details.\");\r\n            this.render_message(\"*API Error. See console logs for details.*\", \"assistant\");\r\n            this.end_stream();\r\n            reject(e);\r\n          });\r\n          this.active_stream.stream();\r\n        } catch (err) {\r\n          console.error(err);\r\n          new Obsidian.Notice(\"Smart Connections Error Streaming Response. See console for details.\");\r\n          this.end_stream();\r\n          reject(err);\r\n        }\r\n      });\r\n      // console.log(full_str);\r\n      await this.render_message(full_str, \"assistant\");\r\n      this.chat.new_message_in_thread({\r\n        role: \"assistant\",\r\n        content: full_str\r\n      });\r\n      return;\r\n    }else{\r\n      try{\r\n        const response = await (0, Obsidian.requestUrl)({\r\n          url: `https://api.openai.com/v1/chat/completions`,\r\n          method: \"POST\",\r\n          headers: {\r\n            Authorization: `Bearer ${this.plugin.settings.api_key}`,\r\n            \"Content-Type\": \"application/json\"\r\n          },\r\n          contentType: \"application/json\",\r\n          body: JSON.stringify(opts),\r\n          throw: false\r\n        });\r\n        // console.log(response);\r\n        return JSON.parse(response.text).choices[0].message.content;\r\n      }catch(err){\r\n        new Obsidian.Notice(`Smart Connections API Error :: ${err}`);\r\n      }\r\n    }\r\n  }\r\n\r\n  end_stream() {\r\n    if(this.active_stream){\r\n      this.active_stream.close();\r\n      this.active_stream = null;\r\n    }\r\n    this.unset_streaming_ux();\r\n    if(this.dotdotdot_interval){\r\n      clearInterval(this.dotdotdot_interval);\r\n      this.dotdotdot_interval = null;\r\n      // remove parent of active_elm\r\n      this.active_elm.parentElement.remove();\r\n      this.active_elm = null;\r\n    }\r\n  }\r\n\r\n  async get_context_hyde(user_input) {\r\n    this.chat.reset_context();\r\n    // count current chat ml messages to determine 'question' or 'chat log' wording\r\n    const hyd_input = `Anticipate what the user is seeking. Respond in the form of a hypothetical note written by the user. The note may contain statements as paragraphs, lists, or checklists in markdown format with no headings. Please respond with one hypothetical note and abstain from any other commentary. Use the format: PARENT FOLDER NAME > CHILD FOLDER NAME > FILE NAME > HEADING 1 > HEADING 2 > HEADING 3: HYPOTHETICAL NOTE CONTENTS.`;\r\n    // complete\r\n    const chatml = [\r\n      {\r\n        role: \"system\",\r\n        content: hyd_input\r\n      },\r\n      {\r\n        role: \"user\",\r\n        content: user_input\r\n      }\r\n    ];\r\n    const hyd = await this.request_chatgpt_completion({\r\n      messages: chatml,\r\n      stream: false,\r\n      temperature: 0,\r\n      max_tokens: 137,\r\n    });\r\n    this.chat.hyd = hyd;\r\n    // console.log(hyd);\r\n    let filter = {};\r\n    // if contains folder reference represented by /folder/\r\n    if(this.chat.contains_folder_reference(user_input)) {\r\n      // get folder references\r\n      const folder_refs = this.chat.get_folder_references(user_input);\r\n      // console.log(folder_refs);\r\n      // if folder references are valid (string or array of strings)\r\n      if(folder_refs){\r\n        filter = {\r\n          path_begins_with: folder_refs\r\n        };\r\n      }\r\n    }\r\n    // search for nearest based on hyd\r\n    let nearest = await this.plugin.api.search(hyd, filter);\r\n    console.log(\"nearest\", nearest.length);\r\n    nearest = this.get_nearest_until_next_dev_exceeds_std_dev(nearest);\r\n    console.log(\"nearest after std dev slice\", nearest.length);\r\n    nearest = this.sort_by_len_adjusted_similarity(nearest);\r\n\r\n    return await this.get_context_for_prompt(nearest);\r\n  }\r\n\r\n\r\n  sort_by_len_adjusted_similarity(nearest) {\r\n    // re-sort by quotient of similarity divided by len DESC\r\n    nearest = nearest.sort((a, b) => {\r\n      const a_score = a.similarity / a.len;\r\n      const b_score = b.similarity / b.len;\r\n      // if a is greater than b, return -1\r\n      if (a_score > b_score)\r\n        return -1;\r\n      // if a is less than b, return 1\r\n      if (a_score < b_score)\r\n        return 1;\r\n      // if a is equal to b, return 0\r\n      return 0;\r\n    });\r\n    return nearest;\r\n  }\r\n\r\n  get_nearest_until_next_dev_exceeds_std_dev(nearest) {\r\n    // get std dev of similarity\r\n    const sim = nearest.map((n) => n.similarity);\r\n    const mean = sim.reduce((a, b) => a + b) / sim.length;\r\n    let std_dev = Math.sqrt(sim.map((x) => Math.pow(x - mean, 2)).reduce((a, b) => a + b) / sim.length);\r\n    // slice where next item deviation is greater than std_dev\r\n    let slice_i = 0;\r\n    while (slice_i < nearest.length) {\r\n      const next = nearest[slice_i + 1];\r\n      if (next) {\r\n        const next_dev = Math.abs(next.similarity - nearest[slice_i].similarity);\r\n        if (next_dev > std_dev) {\r\n          if(slice_i < 3) std_dev = std_dev * 1.5;\r\n          else break;\r\n        }\r\n      }\r\n      slice_i++;\r\n    }\r\n    // select top results\r\n    nearest = nearest.slice(0, slice_i+1);\r\n    return nearest;\r\n  }\r\n  // this.test_get_nearest_until_next_dev_exceeds_std_dev();\r\n  // // test get_nearest_until_next_dev_exceeds_std_dev\r\n  // test_get_nearest_until_next_dev_exceeds_std_dev() {\r\n  //   const nearest = [{similarity: 0.99}, {similarity: 0.98}, {similarity: 0.97}, {similarity: 0.96}, {similarity: 0.95}, {similarity: 0.94}, {similarity: 0.93}, {similarity: 0.92}, {similarity: 0.91}, {similarity: 0.9}, {similarity: 0.79}, {similarity: 0.78}, {similarity: 0.77}, {similarity: 0.76}, {similarity: 0.75}, {similarity: 0.74}, {similarity: 0.73}, {similarity: 0.72}];\r\n  //   const result = this.get_nearest_until_next_dev_exceeds_std_dev(nearest);\r\n  //   if(result.length !== 10){\r\n  //     console.error(\"get_nearest_until_next_dev_exceeds_std_dev failed\", result);\r\n  //   }\r\n  // }\r\n\r\n  async get_context_for_prompt(nearest) {\r\n    let context = [];\r\n    const MAX_SOURCES = (this.plugin.settings.smart_chat_model === 'gpt-4-1106-preview') ? 42 : 20;\r\n    const MAX_CHARS = get_max_chars(this.plugin.settings.smart_chat_model) / 2;\r\n    let char_accum = 0;\r\n    for (let i = 0; i < nearest.length; i++) {\r\n      if (context.length >= MAX_SOURCES)\r\n        break;\r\n      if (char_accum >= MAX_CHARS)\r\n        break;\r\n      if (typeof nearest[i].link !== 'string')\r\n        continue;\r\n      // generate breadcrumbs\r\n      const breadcrumbs = nearest[i].link.replace(/#/g, \" > \").replace(\".md\", \"\").replace(/\\//g, \" > \");\r\n      let new_context = `${breadcrumbs}:\\n`;\r\n      // get max available chars to add to context\r\n      const max_available_chars = MAX_CHARS - char_accum - new_context.length;\r\n      if (nearest[i].link.indexOf(\"#\") !== -1) { // is block\r\n        new_context += await this.plugin.block_retriever(nearest[i].link, { max_chars: max_available_chars });\r\n      } else { // is file\r\n        new_context += await this.plugin.file_retriever(nearest[i].link, { max_chars: max_available_chars });\r\n      }\r\n      // add to char_accum\r\n      char_accum += new_context.length;\r\n      // add to context\r\n      context.push({\r\n        link: nearest[i].link,\r\n        text: new_context\r\n      });\r\n    }\r\n    // context sources\r\n    console.log(\"context sources: \" + context.length);\r\n    // char_accum divided by 4 and rounded to nearest integer for estimated tokens\r\n    console.log(\"total context tokens: ~\" + Math.round(char_accum / 3.5));\r\n    // build context input\r\n    this.chat.context = `Anticipate the type of answer desired by the user. Imagine the following ${context.length} notes were written by the user and contain all the necessary information to answer the user's question. Begin responses with \"${SMART_TRANSLATION[this.plugin.settings.language].prompt}...\"`;\r\n    for(let i = 0; i < context.length; i++) {\r\n      this.chat.context += `\\n---BEGIN #${i+1}---\\n${context[i].text}\\n---END #${i+1}---`;\r\n    }\r\n    return this.chat.context;\r\n  }\r\n\r\n\r\n}\r\n\r\nfunction get_max_chars(model=\"gpt-3.5-turbo\") {\r\n  const MAX_CHAR_MAP = {\r\n    \"gpt-3.5-turbo-16k\": 48000,\r\n    \"gpt-4\": 24000,\r\n    \"gpt-3.5-turbo\": 12000,\r\n    \"gpt-4-1106-preview\": 200000,\r\n  };\r\n  return MAX_CHAR_MAP[model];\r\n}\r\n/**\r\n * SmartConnectionsChatModel\r\n * ---\r\n * - 'thread' format: Array[Array[Object{role, content, hyde}]]\r\n *  - [Turn[variation{}], Turn[variation{}, variation{}], ...]\r\n * - Saves in 'thread' format to JSON file in .smart-connections folder using chat_id as filename\r\n * - Loads chat in 'thread' format Array[Array[Object{role, content, hyde}]] from JSON file in .smart-connections folder\r\n * - prepares chat_ml returns in 'ChatML' format\r\n *  - strips all but role and content properties from Object in ChatML format\r\n * - ChatML Array[Object{role, content}]\r\n *  - [Current_Variation_For_Turn_1{}, Current_Variation_For_Turn_2{}, ...]\r\n */\r\nclass SmartConnectionsChatModel {\r\n  constructor(plugin) {\r\n    this.app = plugin.app;\r\n    this.plugin = plugin;\r\n    this.chat_id = null;\r\n    this.chat_ml = [];\r\n    this.context = null;\r\n    this.hyd = null;\r\n    this.thread = [];\r\n  }\r\n  async save_chat() {\r\n    // return if thread is empty\r\n    if (this.thread.length === 0) return;\r\n    // save chat to file in .smart-connections folder\r\n    // create .smart-connections/chats/ folder if it doesn't exist\r\n    if (!(await this.app.vault.adapter.exists(\".smart-connections/chats\"))) {\r\n      await this.app.vault.adapter.mkdir(\".smart-connections/chats\");\r\n    }\r\n    // if chat_id is not set, set it to UNTITLED-${unix timestamp}\r\n    if (!this.chat_id) {\r\n      this.chat_id = this.name() + \"\u2014\" + this.get_file_date_string();\r\n    }\r\n    // validate chat_id is set to valid filename characters (letters, numbers, underscores, dashes, em dash, and spaces)\r\n    if (!this.chat_id.match(/^[a-zA-Z0-9_\u2014\\- ]+$/)) {\r\n      console.log(\"Invalid chat_id: \" + this.chat_id);\r\n      new Obsidian.Notice(\"[Smart Connections] Failed to save chat. Invalid chat_id: '\" + this.chat_id + \"'\");\r\n    }\r\n    // filename is chat_id\r\n    const chat_file = this.chat_id + \".json\";\r\n    this.app.vault.adapter.write(\r\n      \".smart-connections/chats/\" + chat_file,\r\n      JSON.stringify(this.thread, null, 2)\r\n    );\r\n  }\r\n  async load_chat(chat_id) {\r\n    this.chat_id = chat_id;\r\n    // load chat from file in .smart-connections folder\r\n    // filename is chat_id\r\n    const chat_file = this.chat_id + \".json\";\r\n    // read file\r\n    let chat_json = await this.app.vault.adapter.read(\r\n      \".smart-connections/chats/\" + chat_file\r\n    );\r\n    // parse json\r\n    this.thread = JSON.parse(chat_json);\r\n    // load chat_ml\r\n    this.chat_ml = this.prepare_chat_ml();\r\n    // render messages in chat view\r\n    // for each turn in chat_ml\r\n    // console.log(this.thread);\r\n    // console.log(this.chat_ml);\r\n  }\r\n  // prepare chat_ml from chat\r\n  // gets the last message of each turn unless turn_variation_offsets=[[turn_index,variation_index]] is specified in offset\r\n  prepare_chat_ml(turn_variation_offsets=[]) {\r\n    // if no turn_variation_offsets, get the last message of each turn\r\n    if(turn_variation_offsets.length === 0){\r\n      this.chat_ml = this.thread.map(turn => {\r\n        return turn[turn.length - 1];\r\n      });\r\n    }else{\r\n      // create an array from turn_variation_offsets that indexes variation_index at turn_index\r\n      // ex. [[3,5]] => [undefined, undefined, undefined, 5]\r\n      let turn_variation_index = [];\r\n      for(let i = 0; i < turn_variation_offsets.length; i++){\r\n        turn_variation_index[turn_variation_offsets[i][0]] = turn_variation_offsets[i][1];\r\n      }\r\n      // loop through chat\r\n      this.chat_ml = this.thread.map((turn, turn_index) => {\r\n        // if there is an index for this turn, return the variation at that index\r\n        if(turn_variation_index[turn_index] !== undefined){\r\n          return turn[turn_variation_index[turn_index]];\r\n        }\r\n        // otherwise return the last message of the turn\r\n        return turn[turn.length - 1];\r\n      });\r\n    }\r\n    // strip all but role and content properties from each message\r\n    this.chat_ml = this.chat_ml.map(message => {\r\n      return {\r\n        role: message.role,\r\n        content: message.content\r\n      };\r\n    });\r\n    return this.chat_ml;\r\n  }\r\n  last() {\r\n    // get last message from chat\r\n    return this.thread[this.thread.length - 1][this.thread[this.thread.length - 1].length - 1];\r\n  }\r\n  last_from() {\r\n    return this.last().role;\r\n  }\r\n  // returns user_input or completion\r\n  last_message() {\r\n    return this.last().content;\r\n  }\r\n  // message={}\r\n  // add new message to thread\r\n  new_message_in_thread(message, turn=-1) {\r\n    // if turn is -1, add to new turn\r\n    if(this.context){\r\n      message.context = this.context;\r\n      this.context = null;\r\n    }\r\n    if(this.hyd){\r\n      message.hyd = this.hyd;\r\n      this.hyd = null;\r\n    }\r\n    if (turn === -1) {\r\n      this.thread.push([message]);\r\n    }else{\r\n      // otherwise add to specified turn\r\n      this.thread[turn].push(message);\r\n    }\r\n  }\r\n  reset_context(){\r\n    this.context = null;\r\n    this.hyd = null;\r\n  }\r\n  async rename_chat(new_name){\r\n    // check if current chat_id file exists\r\n    if (this.chat_id && await this.app.vault.adapter.exists(\".smart-connections/chats/\" + this.chat_id + \".json\")) {\r\n      new_name = this.chat_id.replace(this.name(), new_name);\r\n      // rename file if it exists\r\n      await this.app.vault.adapter.rename(\r\n        \".smart-connections/chats/\" + this.chat_id + \".json\",\r\n        \".smart-connections/chats/\" + new_name + \".json\"\r\n      );\r\n      // set chat_id to new_name\r\n      this.chat_id = new_name;\r\n    }else{\r\n      this.chat_id = new_name + \"\u2014\" + this.get_file_date_string();\r\n      // save chat\r\n      await this.save_chat();\r\n    }\r\n\r\n  }\r\n\r\n  name() {\r\n    if(this.chat_id){\r\n      // remove date after last em dash\r\n      return this.chat_id.replace(/\u2014[^\u2014]*$/,\"\");\r\n    }\r\n    return \"UNTITLED\";\r\n  }\r\n\r\n  get_file_date_string() {\r\n    return new Date().toISOString().replace(/(T|:|\\..*)/g, \" \").trim();\r\n  }\r\n  // get response from with note context\r\n  async get_response_with_note_context(user_input, chat_view) {\r\n    let system_input = \"Imagine the following notes were written by the user and contain the necessary information to synthesize a useful answer the user's query:\\n\";\r\n    // extract internal links\r\n    const notes = this.extract_internal_links(user_input);\r\n    // get content of internal links as context\r\n    let max_chars = get_max_chars(this.plugin.settings.smart_chat_model);\r\n    for(let i = 0; i < notes.length; i++){\r\n      // max chars for this note is max_chars divided by number of notes left\r\n      const this_max_chars = (notes.length - i > 1) ? Math.floor(max_chars / (notes.length - i)) : max_chars;\r\n      // console.log(\"file context max chars: \" + this_max_chars);\r\n      const note_content = await this.get_note_contents(notes[i], {char_limit: this_max_chars});\r\n      console.log(note_content);\r\n      system_input += `---BEGIN NOTE: [[${notes[i].basename}]]---\\n`\r\n      system_input += note_content;\r\n      system_input += `---END NOTE---\\n`\r\n      max_chars -= note_content.length;\r\n      if(max_chars <= 0) break;\r\n    }\r\n    this.context = system_input;\r\n    const chatml = [\r\n      {\r\n        role: \"system\",\r\n        content: system_input\r\n      },\r\n      {\r\n        role: \"user\",\r\n        content: user_input\r\n      }\r\n    ];\r\n    chat_view.request_chatgpt_completion({messages: chatml, temperature: 0});\r\n  }\r\n  // check if contains internal link\r\n  contains_internal_link(user_input) {\r\n    if(user_input.indexOf(\"[[\") === -1) return false;\r\n    if(user_input.indexOf(\"]]\") === -1) return false;\r\n    return true;\r\n  }\r\n  // check if contains folder reference (ex. /folder/, or /folder/subfolder/)\r\n  contains_folder_reference(user_input) {\r\n    if(user_input.indexOf(\"/\") === -1) return false;\r\n    if(user_input.indexOf(\"/\") === user_input.lastIndexOf(\"/\")) return false;\r\n    return true;\r\n  }\r\n  // get folder references from user input\r\n  get_folder_references(user_input) {\r\n    // use this.folders to extract folder references by longest first (ex. /folder/subfolder/ before /folder/) to avoid matching /folder/subfolder/ as /folder/\r\n    const folders = this.plugin.folders.slice(); // copy folders array\r\n    const matches = folders.sort((a, b) => b.length - a.length).map(folder => {\r\n      // check if folder is in user_input\r\n      if(user_input.indexOf(folder) !== -1){\r\n        // remove folder from user_input to prevent matching /folder/subfolder/ as /folder/\r\n        user_input = user_input.replace(folder, \"\");\r\n        return folder;\r\n      }\r\n      return false;\r\n    }).filter(folder => folder);\r\n    console.log(matches);\r\n    // return array of matches\r\n    if(matches) return matches;\r\n    return false;\r\n  }\r\n\r\n\r\n  // extract internal links\r\n  extract_internal_links(user_input) {\r\n    const matches = user_input.match(/\\[\\[(.*?)\\]\\]/g);\r\n    console.log(matches);\r\n    // return array of TFile objects\r\n    if(matches) return matches.map(match => {\r\n      return this.app.metadataCache.getFirstLinkpathDest(match.replace(\"[[\", \"\").replace(\"]]\", \"\"), \"/\");\r\n    });\r\n    return [];\r\n  }\r\n  // get context from internal links\r\n  async get_note_contents(note, opts={}) {\r\n    opts = {\r\n      char_limit: 10000,\r\n      ...opts\r\n    }\r\n    // return if note is not a file\r\n    if(!(note instanceof Obsidian.TFile)) return \"\";\r\n    // get file content\r\n    let file_content = await this.app.vault.cachedRead(note);\r\n    //cut off front matter\r\n    if (this.plugin.settings.cut_off_frontmatter) {\r\n      file_content = file_content.replace(/\\s*---[\\s\\S]*?---/,\"\");\r\n    }\r\n    // check if contains dataview code block\r\n    if(file_content.indexOf(\"```dataview\") > -1){\r\n      // if contains dataview code block get all dataview code blocks\r\n      file_content = await this.render_dataview_queries(file_content, note.path, opts);\r\n    }\r\n    return file_content.substring(0, opts.char_limit);\r\n  }\r\n\r\n\r\n  async render_dataview_queries(file_content, note_path, opts={}) {\r\n    opts = {\r\n      char_limit: null,\r\n      ...opts\r\n    };\r\n    // use window to get dataview api\r\n    const dataview_api = window[\"DataviewAPI\"];\r\n    // skip if dataview api not found\r\n    if(!dataview_api) return file_content;\r\n    const dataview_code_blocks = file_content.match(/```dataview(.*?)```/gs);\r\n    // for each dataview code block\r\n    for (let i = 0; i < dataview_code_blocks.length; i++) {\r\n      // if opts char_limit is less than indexOf dataview code block, break\r\n      if(opts.char_limit && opts.char_limit < file_content.indexOf(dataview_code_blocks[i])) break;\r\n      // get dataview code block\r\n      const dataview_code_block = dataview_code_blocks[i];\r\n      // get content of dataview code block\r\n      const dataview_code_block_content = dataview_code_block.replace(\"```dataview\", \"\").replace(\"```\", \"\");\r\n      // get dataview query result\r\n      const dataview_query_result = await dataview_api.queryMarkdown(dataview_code_block_content, note_path, null);\r\n      // if query result is successful, replace dataview code block with query result\r\n      if (dataview_query_result.successful) {\r\n        file_content = file_content.replace(dataview_code_block, dataview_query_result.value);\r\n      }\r\n    }\r\n    return file_content;\r\n  }\r\n}\r\n\r\nclass SmartConnectionsChatHistoryModal extends Obsidian.FuzzySuggestModal {\r\n  constructor(app, view, files) {\r\n    super(app);\r\n    this.app = app;\r\n    this.view = view;\r\n    this.setPlaceholder(\"Type the name of a chat session...\");\r\n  }\r\n  getItems() {\r\n    if (!this.view.files) {\r\n      return [];\r\n    }\r\n    return this.view.files;\r\n  }\r\n  getItemText(item) {\r\n    // if not UNTITLED, remove date after last em dash\r\n    if(item.indexOf(\"UNTITLED\") === -1){\r\n      item.replace(/\u2014[^\u2014]*$/,\"\");\r\n    }\r\n    return item;\r\n  }\r\n  onChooseItem(session) {\r\n    this.view.open_chat(session);\r\n  }\r\n}\r\n\r\n// File Select Fuzzy Suggest Modal\r\nclass SmartConnectionsFileSelectModal extends Obsidian.FuzzySuggestModal {\r\n  constructor(app, view) {\r\n    super(app);\r\n    this.app = app;\r\n    this.view = view;\r\n    this.setPlaceholder(\"Type the name of a file...\");\r\n  }\r\n  getItems() {\r\n    // get all markdown files\r\n    return this.app.vault.getMarkdownFiles().sort((a, b) => a.basename.localeCompare(b.basename));\r\n  }\r\n  getItemText(item) {\r\n    return item.basename;\r\n  }\r\n  onChooseItem(file) {\r\n    this.view.insert_selection(file.basename + \"]] \");\r\n  }\r\n}\r\n// Folder Select Fuzzy Suggest Modal\r\nclass SmartConnectionsFolderSelectModal extends Obsidian.FuzzySuggestModal {\r\n  constructor(app, view) {\r\n    super(app);\r\n    this.app = app;\r\n    this.view = view;\r\n    this.setPlaceholder(\"Type the name of a folder...\");\r\n  }\r\n  getItems() {\r\n    return this.view.plugin.folders;\r\n  }\r\n  getItemText(item) {\r\n    return item;\r\n  }\r\n  onChooseItem(folder) {\r\n    this.view.insert_selection(folder + \"/ \");\r\n  }\r\n}\r\n\r\n\r\n// Handle API response streaming\r\nclass ScStreamer {\r\n  // constructor\r\n  constructor(url, options) {\r\n    // set default options\r\n    options = options || {};\r\n    this.url = url;\r\n    this.method = options.method || 'GET';\r\n    this.headers = options.headers || {};\r\n    this.payload = options.payload || null;\r\n    this.withCredentials = options.withCredentials || false;\r\n    this.listeners = {};\r\n    this.readyState = this.CONNECTING;\r\n    this.progress = 0;\r\n    this.chunk = '';\r\n    this.xhr = null;\r\n    this.FIELD_SEPARATOR = ':';\r\n    this.INITIALIZING = -1;\r\n    this.CONNECTING = 0;\r\n    this.OPEN = 1;\r\n    this.CLOSED = 2;\r\n  }\r\n  // addEventListener\r\n  addEventListener(type, listener) {\r\n    // check if the type is in the listeners\r\n    if (!this.listeners[type]) {\r\n      this.listeners[type] = [];\r\n    }\r\n    // check if the listener is already in the listeners\r\n    if(this.listeners[type].indexOf(listener) === -1) {\r\n      this.listeners[type].push(listener);\r\n    }\r\n  }\r\n  // removeEventListener\r\n  removeEventListener(type, listener) {\r\n    // check if listener type is undefined\r\n    if (!this.listeners[type]) {\r\n      return;\r\n    }\r\n    let filtered = [];\r\n    // loop through the listeners\r\n    for (let i = 0; i < this.listeners[type].length; i++) {\r\n      // check if the listener is the same\r\n      if (this.listeners[type][i] !== listener) {\r\n        filtered.push(this.listeners[type][i]);\r\n      }\r\n    }\r\n    // check if the listeners are empty\r\n    if (this.listeners[type].length === 0) {\r\n      delete this.listeners[type];\r\n    } else {\r\n      this.listeners[type] = filtered;\r\n    }\r\n  }\r\n  // dispatchEvent\r\n  dispatchEvent(event) {\r\n    // if no event return true\r\n    if (!event) {\r\n      return true;\r\n    }\r\n    // set event source to this\r\n    event.source = this;\r\n    // set onHandler to on + event type\r\n    let onHandler = 'on' + event.type;\r\n    // check if the onHandler has own property named same as onHandler\r\n    if (this.hasOwnProperty(onHandler)) {\r\n      // call the onHandler\r\n      this[onHandler].call(this, event);\r\n      // check if the event is default prevented\r\n      if (event.defaultPrevented) {\r\n        return false;\r\n      }\r\n    }\r\n    // check if the event type is in the listeners\r\n    if (this.listeners[event.type]) {\r\n      return this.listeners[event.type].every(function(callback) {\r\n        callback(event);\r\n        return !event.defaultPrevented;\r\n      });\r\n    }\r\n    return true;\r\n  }\r\n  // _setReadyState\r\n  _setReadyState(state) {\r\n    // set event type to readyStateChange\r\n    let event = new CustomEvent('readyStateChange');\r\n    // set event readyState to state\r\n    event.readyState = state;\r\n    // set readyState to state\r\n    this.readyState = state;\r\n    // dispatch event\r\n    this.dispatchEvent(event);\r\n  }\r\n  // _onStreamFailure\r\n  _onStreamFailure(e) {\r\n    // set event type to error\r\n    let event = new CustomEvent('error');\r\n    // set event data to e\r\n    event.data = e.currentTarget.response;\r\n    // dispatch event\r\n    this.dispatchEvent(event);\r\n    this.close();\r\n  }\r\n  // _onStreamAbort\r\n  _onStreamAbort(e) {\r\n    // set to abort\r\n    let event = new CustomEvent('abort');\r\n    // close\r\n    this.close();\r\n  }\r\n  // _onStreamProgress\r\n  _onStreamProgress(e) {\r\n    // if not xhr return\r\n    if (!this.xhr) {\r\n      return;\r\n    }\r\n    // if xhr status is not 200 return\r\n    if (this.xhr.status !== 200) {\r\n      // onStreamFailure\r\n      this._onStreamFailure(e);\r\n      return;\r\n    }\r\n    // if ready state is CONNECTING\r\n    if (this.readyState === this.CONNECTING) {\r\n      // dispatch event\r\n      this.dispatchEvent(new CustomEvent('open'));\r\n      // set ready state to OPEN\r\n      this._setReadyState(this.OPEN);\r\n    }\r\n    // parse the received data.\r\n    let data = this.xhr.responseText.substring(this.progress);\r\n    // update progress\r\n    this.progress += data.length;\r\n    // split the data by new line and parse each line\r\n    data.split(/(\\r\\n|\\r|\\n){2}/g).forEach(function(part){\r\n      if(part.trim().length === 0) {\r\n        this.dispatchEvent(this._parseEventChunk(this.chunk.trim()));\r\n        this.chunk = '';\r\n      } else {\r\n        this.chunk += part;\r\n      }\r\n    }.bind(this));\r\n  }\r\n  // _onStreamLoaded\r\n  _onStreamLoaded(e) {\r\n    this._onStreamProgress(e);\r\n    // parse the last chunk\r\n    this.dispatchEvent(this._parseEventChunk(this.chunk));\r\n    this.chunk = '';\r\n  }\r\n  // _parseEventChunk\r\n  _parseEventChunk(chunk) {\r\n    // if no chunk or chunk is empty return\r\n    if (!chunk || chunk.length === 0) {\r\n      return null;\r\n    }\r\n    // init e\r\n    let e = {id: null, retry: null, data: '', event: 'message'};\r\n    // split the chunk by new line\r\n    chunk.split(/(\\r\\n|\\r|\\n)/).forEach(function(line) {\r\n      line = line.trimRight();\r\n      let index = line.indexOf(this.FIELD_SEPARATOR);\r\n      if(index <= 0) {\r\n        return;\r\n      }\r\n      // field\r\n      let field = line.substring(0, index);\r\n      if(!(field in e)) {\r\n        return;\r\n      }\r\n      // value\r\n      let value = line.substring(index + 1).trimLeft();\r\n      if(field === 'data') {\r\n        e[field] += value;\r\n      } else {\r\n        e[field] = value;\r\n      }\r\n    }.bind(this));\r\n    // return event\r\n    let event = new CustomEvent(e.event);\r\n    event.data = e.data;\r\n    event.id = e.id;\r\n    return event;\r\n  }\r\n  // _checkStreamClosed\r\n  _checkStreamClosed() {\r\n    if(!this.xhr) {\r\n      return;\r\n    }\r\n    if(this.xhr.readyState === XMLHttpRequest.DONE) {\r\n      this._setReadyState(this.CLOSED);\r\n    }\r\n  }\r\n  // stream\r\n  stream() {\r\n    // set ready state to connecting\r\n    this._setReadyState(this.CONNECTING);\r\n    // set xhr to new XMLHttpRequest\r\n    this.xhr = new XMLHttpRequest();\r\n    // set xhr progress to _onStreamProgress\r\n    this.xhr.addEventListener('progress', this._onStreamProgress.bind(this));\r\n    // set xhr load to _onStreamLoaded\r\n    this.xhr.addEventListener('load', this._onStreamLoaded.bind(this));\r\n    // set xhr ready state change to _checkStreamClosed\r\n    this.xhr.addEventListener('readystatechange', this._checkStreamClosed.bind(this));\r\n    // set xhr error to _onStreamFailure\r\n    this.xhr.addEventListener('error', this._onStreamFailure.bind(this));\r\n    // set xhr abort to _onStreamAbort\r\n    this.xhr.addEventListener('abort', this._onStreamAbort.bind(this));\r\n    // open xhr\r\n    this.xhr.open(this.method, this.url);\r\n    // headers to xhr\r\n    for (let header in this.headers) {\r\n      this.xhr.setRequestHeader(header, this.headers[header]);\r\n    }\r\n    // credentials to xhr\r\n    this.xhr.withCredentials = this.withCredentials;\r\n    // send xhr\r\n    this.xhr.send(this.payload);\r\n  }\r\n  // close\r\n  close() {\r\n    if(this.readyState === this.CLOSED) {\r\n      return;\r\n    }\r\n    this.xhr.abort();\r\n    this.xhr = null;\r\n    this._setReadyState(this.CLOSED);\r\n  }\r\n}\r\n\r\nmodule.exports = SmartConnectionsPlugin;"],
  "mappings": ";AAAA,IAAM,WAAW,QAAQ,UAAU;AACnC,IAAM,UAAU,MAAM;AAAA,EACpB,YAAY,QAAQ;AAClB,SAAK,SAAS;AAAA,MACZ,WAAW;AAAA,MACX,aAAa;AAAA,MACb,gBAAgB;AAAA,MAChB,eAAe;AAAA,MACf,cAAc;AAAA,MACd,gBAAgB;AAAA,MAChB,cAAc;AAAA,MACd,eAAe;AAAA,MACf,GAAG;AAAA,IACL;AACA,SAAK,YAAY,KAAK,OAAO;AAC7B,SAAK,cAAc,OAAO;AAC1B,SAAK,YAAY,KAAK,cAAc,MAAM,KAAK;AAC/C,SAAK,aAAa;AAAA,EACpB;AAAA,EACA,MAAM,YAAY,MAAM;AACtB,QAAI,KAAK,OAAO,gBAAgB;AAC9B,aAAO,MAAM,KAAK,OAAO,eAAe,IAAI;AAAA,IAC9C,OAAO;AACL,YAAM,IAAI,MAAM,wBAAwB;AAAA,IAC1C;AAAA,EACF;AAAA,EACA,MAAM,MAAM,MAAM;AAChB,QAAI,KAAK,OAAO,eAAe;AAC7B,aAAO,MAAM,KAAK,OAAO,cAAc,IAAI;AAAA,IAC7C,OAAO;AACL,YAAM,IAAI,MAAM,uBAAuB;AAAA,IACzC;AAAA,EACF;AAAA,EACA,MAAM,UAAU,MAAM;AACpB,QAAI,KAAK,OAAO,cAAc;AAC5B,aAAO,MAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IAC5C,OAAO;AACL,YAAM,IAAI,MAAM,sBAAsB;AAAA,IACxC;AAAA,EACF;AAAA,EACA,MAAM,OAAO,UAAU,UAAU;AAC/B,QAAI,KAAK,OAAO,gBAAgB;AAC9B,aAAO,MAAM,KAAK,OAAO,eAAe,UAAU,QAAQ;AAAA,IAC5D,OAAO;AACL,YAAM,IAAI,MAAM,wBAAwB;AAAA,IAC1C;AAAA,EACF;AAAA,EACA,MAAM,KAAK,MAAM;AACf,QAAI,KAAK,OAAO,cAAc;AAC5B,aAAO,MAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IAC5C,OAAO;AACL,YAAM,IAAI,MAAM,sBAAsB;AAAA,IACxC;AAAA,EACF;AAAA,EACA,MAAM,WAAW,MAAM,MAAM;AAC3B,QAAI,KAAK,OAAO,eAAe;AAC7B,aAAO,MAAM,KAAK,OAAO,cAAc,MAAM,IAAI;AAAA,IACnD,OAAO;AACL,YAAM,IAAI,MAAM,uBAAuB;AAAA,IACzC;AAAA,EACF;AAAA,EACA,MAAM,KAAK,UAAU,GAAG;AACtB,QAAI;AACF,YAAM,kBAAkB,MAAM,KAAK,UAAU,KAAK,SAAS;AAC3D,WAAK,aAAa,KAAK,MAAM,eAAe;AAC5C,cAAQ,IAAI,6BAA6B,KAAK,SAAS;AACvD,aAAO;AAAA,IACT,SAAS,OAAP;AACA,UAAI,UAAU,GAAG;AACf,gBAAQ,IAAI,iBAAiB;AAC7B,cAAM,IAAI,QAAQ,CAAC,MAAM,WAAW,GAAG,MAAM,MAAM,OAAO,CAAC;AAC3D,eAAO,MAAM,KAAK,KAAK,UAAU,CAAC;AAAA,MACpC;AACA,cAAQ;AAAA,QACN;AAAA,MACF;AACA,aAAO;AAAA,IACT;AAAA,EACF;AAAA,EACA,MAAM,uBAAuB;AAC3B,QAAI,CAAE,MAAM,KAAK,YAAY,KAAK,WAAW,GAAI;AAC/C,YAAM,KAAK,MAAM,KAAK,WAAW;AACjC,cAAQ,IAAI,qBAAqB,KAAK,WAAW;AAAA,IACnD,OAAO;AACL,cAAQ,IAAI,4BAA4B,KAAK,WAAW;AAAA,IAC1D;AACA,QAAI,CAAE,MAAM,KAAK,YAAY,KAAK,SAAS,GAAI;AAC7C,YAAM,KAAK,WAAW,KAAK,WAAW,IAAI;AAC1C,cAAQ,IAAI,8BAA8B,KAAK,SAAS;AAAA,IAC1D,OAAO;AACL,cAAQ,IAAI,qCAAqC,KAAK,SAAS;AAAA,IACjE;AAAA,EACF;AAAA,EACA,MAAM,OAAO;AACX,UAAM,aAAa,KAAK,UAAU,KAAK,UAAU;AACjD,UAAM,yBAAyB,MAAM,KAAK,YAAY,KAAK,SAAS;AACpE,QAAI,wBAAwB;AAC1B,YAAM,gBAAgB,WAAW;AACjC,YAAM,qBAAqB,MAAM,KAAK,KAAK,KAAK,SAAS,EAAE;AAAA,QACzD,CAAC,SAAS,KAAK;AAAA,MACjB;AACA,UAAI,gBAAgB,qBAAqB,KAAK;AAC5C,cAAM,KAAK,WAAW,KAAK,WAAW,UAAU;AAChD,gBAAQ,IAAI,2BAA2B,gBAAgB,QAAQ;AAAA,MACjE,OAAO;AACL,cAAM,kBAAkB;AAAA,UACtB;AAAA,UACA;AAAA,UACA,oBAAoB,gBAAgB;AAAA,UACpC,yBAAyB,qBAAqB;AAAA,UAC9C;AAAA,QACF;AACA,gBAAQ,IAAI,gBAAgB,KAAK,GAAG,CAAC;AACrC,cAAM,KAAK;AAAA,UACT,KAAK,cAAc;AAAA,UACnB;AAAA,QACF;AACA,cAAM,IAAI;AAAA,UACR;AAAA,QACF;AAAA,MACF;AAAA,IACF,OAAO;AACL,YAAM,KAAK,qBAAqB;AAChC,aAAO,MAAM,KAAK,KAAK;AAAA,IACzB;AACA,WAAO;AAAA,EACT;AAAA,EACA,QAAQ,SAAS,SAAS;AACxB,QAAI,aAAa;AACjB,QAAI,QAAQ;AACZ,QAAI,QAAQ;AACZ,aAAS,IAAI,GAAG,IAAI,QAAQ,QAAQ,KAAK;AACvC,oBAAc,QAAQ,CAAC,IAAI,QAAQ,CAAC;AACpC,eAAS,QAAQ,CAAC,IAAI,QAAQ,CAAC;AAC/B,eAAS,QAAQ,CAAC,IAAI,QAAQ,CAAC;AAAA,IACjC;AACA,QAAI,UAAU,KAAK,UAAU,GAAG;AAC9B,aAAO;AAAA,IACT,OAAO;AACL,aAAO,cAAc,KAAK,KAAK,KAAK,IAAI,KAAK,KAAK,KAAK;AAAA,IACzD;AAAA,EACF;AAAA,EACA,QAAQ,QAAQ,SAAS,CAAC,GAAG;AAC3B,aAAS;AAAA,MACP,eAAe;AAAA,MACf,GAAG;AAAA,IACL;AACA,QAAI,UAAU,CAAC;AACf,UAAM,YAAY,OAAO,KAAK,KAAK,UAAU;AAC7C,aAAS,IAAI,GAAG,IAAI,UAAU,QAAQ,KAAK;AACzC,UAAI,OAAO,eAAe;AACxB,cAAM,YAAY,KAAK,WAAW,UAAU,CAAC,CAAC,EAAE,KAAK;AACrD,YAAI,UAAU,QAAQ,GAAG,IAAI;AAAI;AAAA,MACnC;AACA,UAAI,OAAO,UAAU;AACnB,YAAI,OAAO,aAAa,UAAU,CAAC;AAAG;AACtC,YAAI,OAAO,aAAa,KAAK,WAAW,UAAU,CAAC,CAAC,EAAE,KAAK;AACzD;AAAA,MACJ;AACA,UAAI,OAAO,kBAAkB;AAC3B,YACE,OAAO,OAAO,qBAAqB,YACnC,CAAC,KAAK,WAAW,UAAU,CAAC,CAAC,EAAE,KAAK,KAAK;AAAA,UACvC,OAAO;AAAA,QACT;AAEA;AACF,YACE,MAAM,QAAQ,OAAO,gBAAgB,KACrC,CAAC,OAAO,iBAAiB;AAAA,UAAK,CAAC,SAC7B,KAAK,WAAW,UAAU,CAAC,CAAC,EAAE,KAAK,KAAK,WAAW,IAAI;AAAA,QACzD;AAEA;AAAA,MACJ;AACA,cAAQ,KAAK;AAAA,QACX,MAAM,KAAK,WAAW,UAAU,CAAC,CAAC,EAAE,KAAK;AAAA,QACzC,YAAY,KAAK,QAAQ,QAAQ,KAAK,WAAW,UAAU,CAAC,CAAC,EAAE,GAAG;AAAA,QAClE,MAAM,KAAK,WAAW,UAAU,CAAC,CAAC,EAAE,KAAK;AAAA,MAC3C,CAAC;AAAA,IACH;AACA,YAAQ,KAAK,SAAU,GAAG,GAAG;AAC3B,aAAO,EAAE,aAAa,EAAE;AAAA,IAC1B,CAAC;AACD,cAAU,QAAQ,MAAM,GAAG,OAAO,aAAa;AAC/C,WAAO;AAAA,EACT;AAAA,EACA,wBAAwB,QAAQ,SAAS,CAAC,GAAG;AAC3C,UAAM,iBAAiB;AAAA,MACrB,KAAK,KAAK;AAAA,IACZ;AACA,aAAS,EAAE,GAAG,gBAAgB,GAAG,OAAO;AACxC,QAAI,MAAM,QAAQ,MAAM,KAAK,OAAO,WAAW,KAAK,SAAS;AAC3D,WAAK,UAAU,CAAC;AAChB,eAAS,IAAI,GAAG,IAAI,OAAO,QAAQ,KAAK;AACtC,aAAK,wBAAwB,OAAO,CAAC,GAAG;AAAA,UACtC,KAAK,KAAK,MAAM,OAAO,MAAM,OAAO,MAAM;AAAA,QAC5C,CAAC;AAAA,MACH;AAAA,IACF,OAAO;AACL,YAAM,YAAY,OAAO,KAAK,KAAK,UAAU;AAC7C,eAAS,IAAI,GAAG,IAAI,UAAU,QAAQ,KAAK;AACzC,YAAI,KAAK,cAAc,KAAK,WAAW,UAAU,CAAC,CAAC,CAAC;AAAG;AACvD,cAAM,MAAM,KAAK;AAAA,UACf;AAAA,UACA,KAAK,WAAW,UAAU,CAAC,CAAC,EAAE;AAAA,QAChC;AACA,YAAI,KAAK,QAAQ,UAAU,CAAC,CAAC,GAAG;AAC9B,eAAK,QAAQ,UAAU,CAAC,CAAC,KAAK;AAAA,QAChC,OAAO;AACL,eAAK,QAAQ,UAAU,CAAC,CAAC,IAAI;AAAA,QAC/B;AAAA,MACF;AAAA,IACF;AACA,QAAI,UAAU,OAAO,KAAK,KAAK,OAAO,EAAE,IAAI,CAAC,QAAQ;AACnD,aAAO;AAAA,QACL;AAAA,QACA,YAAY,KAAK,QAAQ,GAAG;AAAA,MAC9B;AAAA,IACF,CAAC;AACD,cAAU,KAAK,mBAAmB,OAAO;AACzC,cAAU,QAAQ,MAAM,GAAG,OAAO,GAAG;AACrC,cAAU,QAAQ,IAAI,CAAC,SAAS;AAC9B,aAAO;AAAA,QACL,MAAM,KAAK,WAAW,KAAK,GAAG,EAAE,KAAK;AAAA,QACrC,YAAY,KAAK;AAAA,QACjB,KACE,KAAK,WAAW,KAAK,GAAG,EAAE,KAAK,OAC/B,KAAK,WAAW,KAAK,GAAG,EAAE,KAAK;AAAA,MACnC;AAAA,IACF,CAAC;AACD,WAAO;AAAA,EACT;AAAA,EACA,mBAAmB,SAAS;AAC1B,WAAO,QAAQ,KAAK,SAAU,GAAG,GAAG;AAClC,YAAM,UAAU,EAAE;AAClB,YAAM,UAAU,EAAE;AAClB,UAAI,UAAU;AAAS,eAAO;AAC9B,UAAI,UAAU;AAAS,eAAO;AAC9B,aAAO;AAAA,IACT,CAAC;AAAA,EACH;AAAA;AAAA,EAEA,oBAAoB,OAAO;AACzB,YAAQ,IAAI,wBAAwB;AACpC,UAAM,OAAO,OAAO,KAAK,KAAK,UAAU;AACxC,QAAI,qBAAqB;AACzB,eAAW,OAAO,MAAM;AACtB,YAAM,OAAO,KAAK,WAAW,GAAG,EAAE,KAAK;AACvC,UAAI,CAAC,MAAM,KAAK,CAAC,SAAS,KAAK,WAAW,KAAK,IAAI,CAAC,GAAG;AACrD,eAAO,KAAK,WAAW,GAAG;AAC1B;AACA;AAAA,MACF;AACA,UAAI,KAAK,QAAQ,GAAG,IAAI,IAAI;AAC1B,cAAM,aAAa,KAAK,WAAW,GAAG,EAAE,KAAK;AAC7C,YAAI,CAAC,KAAK,WAAW,UAAU,GAAG;AAChC,iBAAO,KAAK,WAAW,GAAG;AAC1B;AACA;AAAA,QACF;AACA,YAAI,CAAC,KAAK,WAAW,UAAU,EAAE,MAAM;AACrC,iBAAO,KAAK,WAAW,GAAG;AAC1B;AACA;AAAA,QACF;AACA,YACE,KAAK,WAAW,UAAU,EAAE,KAAK,YACjC,KAAK,WAAW,UAAU,EAAE,KAAK,SAAS,QAAQ,GAAG,IAAI,GACzD;AACA,iBAAO,KAAK,WAAW,GAAG;AAC1B;AACA;AAAA,QACF;AAAA,MACF;AAAA,IACF;AACA,WAAO,EAAE,oBAAoB,kBAAkB,KAAK,OAAO;AAAA,EAC7D;AAAA,EACA,IAAI,KAAK;AACP,WAAO,KAAK,WAAW,GAAG,KAAK;AAAA,EACjC;AAAA,EACA,SAAS,KAAK;AACZ,UAAM,YAAY,KAAK,IAAI,GAAG;AAC9B,QAAI,aAAa,UAAU,MAAM;AAC/B,aAAO,UAAU;AAAA,IACnB;AACA,WAAO;AAAA,EACT;AAAA,EACA,UAAU,KAAK;AACb,UAAM,OAAO,KAAK,SAAS,GAAG;AAC9B,QAAI,QAAQ,KAAK,OAAO;AACtB,aAAO,KAAK;AAAA,IACd;AACA,WAAO;AAAA,EACT;AAAA,EACA,SAAS,KAAK;AACZ,UAAM,OAAO,KAAK,SAAS,GAAG;AAC9B,QAAI,QAAQ,KAAK,MAAM;AACrB,aAAO,KAAK;AAAA,IACd;AACA,WAAO;AAAA,EACT;AAAA,EACA,SAAS,KAAK;AACZ,UAAM,OAAO,KAAK,SAAS,GAAG;AAC9B,QAAI,QAAQ,KAAK,MAAM;AACrB,aAAO,KAAK;AAAA,IACd;AACA,WAAO;AAAA,EACT;AAAA,EACA,aAAa,KAAK;AAChB,UAAM,OAAO,KAAK,SAAS,GAAG;AAC9B,QAAI,QAAQ,KAAK,UAAU;AACzB,aAAO,KAAK;AAAA,IACd;AACA,WAAO;AAAA,EACT;AAAA,EACA,QAAQ,KAAK;AACX,UAAM,YAAY,KAAK,IAAI,GAAG;AAC9B,QAAI,aAAa,UAAU,KAAK;AAC9B,aAAO,UAAU;AAAA,IACnB;AACA,WAAO;AAAA,EACT;AAAA,EACA,eAAe,KAAK,KAAK,MAAM;AAC7B,SAAK,WAAW,GAAG,IAAI;AAAA,MACrB;AAAA,MACA;AAAA,IACF;AAAA,EACF;AAAA,EACA,iBAAiB,KAAK,cAAc;AAClC,UAAM,QAAQ,KAAK,UAAU,GAAG;AAChC,QAAI,SAAS,SAAS,cAAc;AAClC,aAAO;AAAA,IACT;AACA,WAAO;AAAA,EACT;AAAA,EACA,MAAM,gBAAgB;AACpB,SAAK,aAAa;AAClB,SAAK,aAAa,CAAC;AACnB,QAAI,mBAAmB,KAAK,MAAM,KAAK,IAAI,IAAI,GAAG;AAClD,UAAM,KAAK;AAAA,MACT,KAAK;AAAA,MACL,KAAK,cAAc,iBAAiB,mBAAmB;AAAA,IACzD;AACA,UAAM,KAAK,qBAAqB;AAAA,EAClC;AACF;AAIA,IAAM,mBAAmB;AAAA,EACvB,SAAS;AAAA,EACT,WAAW;AAAA,EACX,iBAAiB;AAAA,EACjB,mBAAmB;AAAA,EACnB,mBAAmB;AAAA,EACnB,WAAW;AAAA,EACX,gBAAgB;AAAA,EAChB,qBAAqB;AAAA,EACrB,eAAe;AAAA,EACf,uBAAuB;AAAA,EACvB,UAAU;AAAA,EACV,YAAY;AAAA,EACZ,kBAAkB;AAAA,EAClB,4BAA4B;AAAA,EAC5B,eAAe;AAAA,EACf,kBAAkB;AAAA,EAClB,WAAW;AAAA,EACX,SAAS;AAAA,EACT,kBAAkB;AACpB;AACA,IAAM,0BAA0B;AAEhC,IAAI;AACJ,IAAM,uBAAuB,CAAC,MAAM,QAAQ;AAI5C,IAAM,oBAAoB;AAAA,EACxB,MAAM;AAAA,IACJ,WAAW,CAAC,MAAM,KAAK,MAAM,QAAQ,OAAO,QAAQ,MAAM,IAAI;AAAA,IAC9D,UAAU;AAAA,IACV,mBAAmB;AAAA,IACnB,mBAAmB;AAAA,EACrB;AAAA,EACA,MAAM;AAAA,IACJ,WAAW,CAAC,MAAM,MAAM,SAAM,OAAI;AAAA,IAClC,UAAU;AAAA,IACV,mBAAmB;AAAA,IACnB,mBAAmB;AAAA,EACrB;AAAA,EACA,MAAM;AAAA,IACJ,WAAW,CAAC,MAAM,OAAO,MAAM,OAAO,OAAO,QAAQ,SAAS,OAAO,MAAM,MAAM,IAAI;AAAA,IACrF,UAAU;AAAA,IACV,mBAAmB;AAAA,IACnB,mBAAmB;AAAA,EACrB;AAAA,EACA,MAAM;AAAA,IACJ,WAAW,CAAC,QAAQ,SAAS,UAAU,UAAU,UAAU,OAAO,OAAO,SAAS,WAAW,WAAW,SAAS;AAAA,IACjH,UAAU;AAAA,IACV,mBAAmB;AAAA,IACnB,mBAAmB;AAAA,EACrB;AAAA,EACA,MAAM;AAAA,IACJ,WAAW,CAAC,OAAO,OAAO,QAAQ,OAAO,OAAO,UAAU,UAAU,UAAU,QAAQ;AAAA,IACtF,UAAU;AAAA,IACV,mBAAmB;AAAA,IACnB,mBAAmB;AAAA,EACrB;AACF;AAGA,IAAM,SAAS,QAAQ,QAAQ;AAE/B,SAAS,IAAI,KAAK;AAChB,SAAO,OAAO,WAAW,KAAK,EAAE,OAAO,GAAG,EAAE,OAAO,KAAK;AAC1D;AAEA,IAAM,yBAAN,cAAqC,SAAS,OAAO;AAAA;AAAA,EAEnD,cAAc;AACZ,UAAM,GAAG,SAAS;AAClB,SAAK,MAAM;AACX,SAAK,oBAAoB;AACzB,SAAK,kBAAkB,CAAC;AACxB,SAAK,UAAU,CAAC;AAChB,SAAK,qBAAqB;AAC1B,SAAK,oBAAoB,CAAC;AAC1B,SAAK,gBAAgB,CAAC;AACtB,SAAK,YAAY,CAAC;AAClB,SAAK,aAAa,CAAC;AACnB,SAAK,WAAW,qBAAqB;AACrC,SAAK,WAAW,kBAAkB,CAAC;AACnC,SAAK,WAAW,oBAAoB,CAAC;AACrC,SAAK,WAAW,QAAQ,CAAC;AACzB,SAAK,WAAW,iBAAiB;AACjC,SAAK,WAAW,oBAAoB,CAAC;AACrC,SAAK,WAAW,cAAc;AAC9B,SAAK,WAAW,wBAAwB;AACxC,SAAK,uBAAuB;AAC5B,SAAK,eAAe;AACpB,SAAK,cAAc,CAAC;AACpB,SAAK,oBAAoB;AACzB,SAAK,mBAAmB;AAAA,EAC1B;AAAA,EAEA,MAAM,SAAS;AAEb,SAAK,IAAI,UAAU,cAAc,KAAK,WAAW,KAAK,IAAI,CAAC;AAAA,EAC7D;AAAA,EACA,WAAW;AACT,SAAK,kBAAkB;AACvB,YAAQ,IAAI,kBAAkB;AAC9B,SAAK,IAAI,UAAU,mBAAmB,2BAA2B;AACjE,SAAK,IAAI,UAAU,mBAAmB,gCAAgC;AAAA,EACxE;AAAA,EACA,MAAM,aAAa;AACjB,YAAQ,IAAI,kCAAkC;AAC9C,cAAU,KAAK,SAAS;AAGxB,UAAM,KAAK,aAAa;AAExB,eAAW,KAAK,iBAAiB,KAAK,IAAI,GAAG,GAAI;AAEjD,gBAAY,KAAK,iBAAiB,KAAK,IAAI,GAAG,KAAQ;AAEtD,SAAK,QAAQ;AACb,SAAK,WAAW;AAAA,MACd,IAAI;AAAA,MACJ,MAAM;AAAA,MACN,MAAM;AAAA,MACN,SAAS,CAAC;AAAA;AAAA,MAEV,gBAAgB,OAAO,WAAW;AAChC,YAAG,OAAO,kBAAkB,GAAG;AAE7B,cAAI,gBAAgB,OAAO,aAAa;AAExC,gBAAM,KAAK,iBAAiB,aAAa;AAAA,QAC3C,OAAO;AAEL,eAAK,gBAAgB,CAAC;AAEtB,gBAAM,KAAK,iBAAiB;AAAA,QAC9B;AAAA,MACF;AAAA,IACF,CAAC;AACD,SAAK,WAAW;AAAA,MACd,IAAI;AAAA,MACJ,MAAM;AAAA,MACN,UAAU,MAAM;AACd,aAAK,UAAU;AAAA,MACjB;AAAA,IACF,CAAC;AAED,SAAK,WAAW;AAAA,MACd,IAAI;AAAA,MACJ,MAAM;AAAA,MACN,UAAU,MAAM;AACd,aAAK,UAAU;AAAA,MACjB;AAAA,IACF,CAAC;AAED,SAAK,WAAW;AAAA,MACd,IAAI;AAAA,MACJ,MAAM;AAAA,MACN,UAAU,MAAM;AACd,aAAK,iBAAiB;AAAA,MACxB;AAAA,IACF,CAAC;AAED,SAAK,cAAc,IAAI,4BAA4B,KAAK,KAAK,IAAI,CAAC;AAElE,SAAK,aAAa,6BAA6B,CAAC,SAAU,IAAI,qBAAqB,MAAM,IAAI,CAAE;AAE/F,SAAK,aAAa,kCAAkC,CAAC,SAAU,IAAI,yBAAyB,MAAM,IAAI,CAAE;AAExG,SAAK,mCAAmC,qBAAqB,KAAK,kBAAkB,KAAK,IAAI,CAAC;AAG9F,QAAG,KAAK,SAAS,WAAW;AAC1B,WAAK,UAAU;AAAA,IACjB;AAEA,QAAG,KAAK,SAAS,WAAW;AAC1B,WAAK,UAAU;AAAA,IACjB;AAEA,QAAG,KAAK,SAAS,YAAY,SAAS;AAEpC,WAAK,SAAS,UAAU;AAExB,YAAM,KAAK,aAAa;AAExB,WAAK,UAAU;AAAA,IACjB;AAEA,SAAK,iBAAiB;AAMtB,SAAK,MAAM,IAAI,YAAY,KAAK,KAAK,IAAI;AAEzC,KAAC,OAAO,gBAAgB,IAAI,KAAK,QAAQ,KAAK,SAAS,MAAM,OAAO,OAAO,gBAAgB,CAAC;AAAA,EAE9F;AAAA,EAEA,MAAM,YAAY;AAChB,SAAK,iBAAiB,IAAI,QAAQ;AAAA,MAChC,aAAa;AAAA,MACb,gBAAgB,KAAK,IAAI,MAAM,QAAQ,OAAO,KAAK,KAAK,IAAI,MAAM,OAAO;AAAA,MACzE,eAAe,KAAK,IAAI,MAAM,QAAQ,MAAM,KAAK,KAAK,IAAI,MAAM,OAAO;AAAA,MACvE,cAAc,KAAK,IAAI,MAAM,QAAQ,KAAK,KAAK,KAAK,IAAI,MAAM,OAAO;AAAA,MACrE,gBAAgB,KAAK,IAAI,MAAM,QAAQ,OAAO,KAAK,KAAK,IAAI,MAAM,OAAO;AAAA,MACzE,cAAc,KAAK,IAAI,MAAM,QAAQ,KAAK,KAAK,KAAK,IAAI,MAAM,OAAO;AAAA,MACrE,eAAe,KAAK,IAAI,MAAM,QAAQ,MAAM,KAAK,KAAK,IAAI,MAAM,OAAO;AAAA,IACzE,CAAC;AACD,SAAK,oBAAoB,MAAM,KAAK,eAAe,KAAK;AACxD,WAAO,KAAK;AAAA,EACd;AAAA,EACA,MAAM,eAAe;AAEnB,QAAG,CAAC,KAAK,SAAS;AAAa,aAAO,IAAI,SAAS,OAAO,2EAA2E;AAErI,UAAM,KAAK,OAAO,GAAG,SAAS,YAAY;AAAA,MACxC,KAAK;AAAA,MACL,QAAQ;AAAA,MACR,SAAS;AAAA,QACP,gBAAgB;AAAA,MAClB;AAAA,MACA,MAAM,KAAK,UAAU;AAAA,QACnB,aAAa,KAAK,SAAS;AAAA,MAC7B,CAAC;AAAA,IACH,CAAC;AACD,QAAG,GAAG,WAAW;AAAK,aAAO,QAAQ,MAAM,+BAA+B,EAAE;AAC5E,YAAQ,IAAI,EAAE;AACd,UAAM,KAAK,IAAI,MAAM,QAAQ,MAAM,+CAA+C,GAAG,KAAK,IAAI;AAC9F,UAAM,KAAK,IAAI,MAAM,QAAQ,MAAM,qDAAqD,GAAG,KAAK,QAAQ;AACxG,UAAM,KAAK,IAAI,MAAM,QAAQ,MAAM,kDAAkD,GAAG,KAAK,MAAM;AACnG,WAAO,iBAAiB,OAAO,OAAO;AACpC,cAAQ,IAAI,qBAAqB,EAAE;AACnC,YAAM,OAAO,IAAI,QAAQ,cAAc,EAAE;AACzC,YAAM,OAAO,IAAI,QAAQ,aAAa,EAAE;AACxC,cAAQ,IAAI,oBAAoB,EAAE;AAAA,IACpC;AACA,WAAO,eAAe,KAAK,SAAS,EAAE;AAAA,EACxC;AAAA,EAEA,MAAM,eAAe;AACnB,SAAK,WAAW,OAAO,OAAO,CAAC,GAAG,kBAAkB,MAAM,KAAK,SAAS,CAAC;AAEzE,QAAG,KAAK,SAAS,mBAAmB,KAAK,SAAS,gBAAgB,SAAS,GAAG;AAE5E,WAAK,kBAAkB,KAAK,SAAS,gBAAgB,MAAM,SAAS,EACjE,OAAO,CAAC,SAAS,KAAK,SAAS,CAAC,EAChC,IAAI,CAAC,SAAS;AACf,eAAO,KAAK,KAAK;AAAA,MACnB,CAAC;AAAA,IACH;AAEA,QAAG,KAAK,SAAS,qBAAqB,KAAK,SAAS,kBAAkB,SAAS,GAAG;AAEhF,YAAM,oBAAoB,KAAK,SAAS,kBACrC,MAAM,SAAS,EACf,OAAO,CAAC,SAAS,KAAK,SAAS,CAAC,EAChC,IAAI,CAAC,WAAW;AACf,iBAAS,OAAO,KAAK;AACrB,eAAO,OAAO,MAAM,EAAE,MAAM,MAAM,SAAS,GAAG;AAAA,MAChD,CAAC;AAEH,WAAK,kBAAkB,KAAK,gBAAgB,OAAO,iBAAiB;AAAA,IACtE;AAEA,QAAG,KAAK,SAAS,qBAAqB,KAAK,SAAS,kBAAkB,SAAS,GAAG;AAChF,WAAK,oBAAoB,KAAK,SAAS,kBACpC,MAAM,SAAS,EACf,OAAO,CAAC,SAAS,KAAK,SAAS,CAAC,EAChC,IAAI,CAAC,WAAW;AACf,eAAO,OAAO,KAAK;AAAA,MACrB,CAAC;AAAA,IACL;AAEA,QAAG,KAAK,SAAS,aAAa,KAAK,SAAS,UAAU,SAAS,GAAG;AAChE,WAAK,YAAY,KAAK,SAAS,UAC5B,MAAM,SAAS,EACf,OAAO,CAAC,SAAS,KAAK,SAAS,CAAC,EAChC,IAAI,CAAC,SAAS;AACb,eAAO,KAAK,KAAK;AAAA,MACnB,CAAC;AAAA,IACL;AAEA,SAAK,oBAAoB,IAAI,OAAO,OAAO,kBAAkB,KAAK,SAAS,QAAQ,EAAE,QAAQ,KAAK,GAAG,SAAS,IAAI;AAElH,UAAM,KAAK,kBAAkB;AAAA,EAC/B;AAAA,EACA,MAAM,aAAa,WAAS,OAAO;AACjC,UAAM,KAAK,SAAS,KAAK,QAAQ;AAEjC,UAAM,KAAK,aAAa;AAExB,QAAG,UAAU;AACX,WAAK,gBAAgB,CAAC;AACtB,YAAM,KAAK,iBAAiB;AAAA,IAC9B;AAAA,EACF;AAAA;AAAA,EAGA,MAAM,mBAAmB;AAEvB,QAAI;AAEF,YAAM,WAAW,OAAO,GAAG,SAAS,YAAY;AAAA,QAC9C,KAAK;AAAA,QACL,QAAQ;AAAA,QACR,SAAS;AAAA,UACP,gBAAgB;AAAA,QAClB;AAAA,QACA,aAAa;AAAA,MACf,CAAC;AAED,YAAM,iBAAiB,KAAK,MAAM,SAAS,IAAI,EAAE;AAGjD,UAAG,mBAAmB,SAAS;AAC7B,YAAI,SAAS,OAAO,qDAAqD,iBAAiB;AAC1F,aAAK,mBAAmB;AACxB,aAAK,aAAa,KAAK;AAAA,MACzB;AAAA,IACF,SAAS,OAAP;AACA,cAAQ,IAAI,KAAK;AAAA,IACnB;AAAA,EACF;AAAA,EAEA,MAAM,kBAAkB,UAAU,WAAW,KAAK;AAChD,QAAI;AACJ,QAAG,SAAS,KAAK,EAAE,SAAS,GAAG;AAC7B,gBAAU,MAAM,KAAK,IAAI,OAAO,QAAQ;AAAA,IAC1C,OAAO;AAEL,cAAQ,IAAI,GAAG;AACf,YAAM,OAAO,KAAK,IAAI,MAAM,sBAAsB,IAAI,UAAU;AAChE,gBAAU,MAAM,KAAK,sBAAsB,IAAI;AAAA,IACjD;AACA,QAAI,QAAQ,QAAQ;AAClB,WAAK,eAAe,WAAW,OAAO;AAAA,IACxC;AAAA,EACF;AAAA,EAEA,MAAM,iBAAiB,gBAAc,MAAM;AACzC,QAAI,OAAO,KAAK,SAAS;AACzB,QAAI,CAAC,MAAM;AAET,YAAM,KAAK,UAAU;AACrB,aAAO,KAAK,SAAS;AAAA,IACvB;AACA,UAAM,KAAK,mBAAmB,aAAa;AAAA,EAC7C;AAAA,EAEA,UAAS;AACP,aAAS,QAAQ,qBAAqB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,wDAMc;AAAA,EACtD;AAAA;AAAA,EAGA,MAAM,mBAAmB;AACvB,UAAM,YAAY,KAAK,IAAI,UAAU,cAAc;AACnD,UAAM,WAAW,IAAI,UAAU,IAAI;AAEnC,QAAG,OAAO,KAAK,cAAc,QAAQ,MAAM,aAAa;AACtD,UAAI,SAAS,OAAO,uFAAuF;AAC3G;AAAA,IACF;AAEA,UAAM,OAAO,KAAK,MAAM,KAAK,OAAO,IAAI,KAAK,cAAc,QAAQ,EAAE,SAAO,CAAC;AAC7E,UAAM,cAAc,KAAK,cAAc,QAAQ,EAAE,IAAI;AAErD,SAAK,UAAU,WAAW;AAAA,EAC5B;AAAA,EAEA,MAAM,YAAY;AAChB,QAAG,KAAK,SAAS,GAAE;AACjB,cAAQ,IAAI,qCAAqC;AACjD;AAAA,IACF;AACA,SAAK,IAAI,UAAU,mBAAmB,2BAA2B;AACjE,UAAM,KAAK,IAAI,UAAU,aAAa,KAAK,EAAE,aAAa;AAAA,MACxD,MAAM;AAAA,MACN,QAAQ;AAAA,IACV,CAAC;AACD,SAAK,IAAI,UAAU;AAAA,MACjB,KAAK,IAAI,UAAU,gBAAgB,2BAA2B,EAAE,CAAC;AAAA,IACnE;AAAA,EACF;AAAA;AAAA,EAEA,WAAW;AACT,aAAS,QAAQ,KAAK,IAAI,UAAU,gBAAgB,2BAA2B,GAAG;AAChF,UAAI,KAAK,gBAAgB,sBAAsB;AAC7C,eAAO,KAAK;AAAA,MACd;AAAA,IACF;AAAA,EACF;AAAA;AAAA,EAEA,MAAM,UAAU,UAAQ,GAAG;AACzB,QAAG,CAAC,KAAK,mBAAmB;AAC1B,cAAQ,IAAI,2BAA2B;AACvC,UAAG,UAAU,GAAG;AAEd,mBAAW,MAAM;AACf,eAAK,UAAU,UAAQ,CAAC;AAAA,QAC1B,GAAG,OAAQ,UAAQ,EAAE;AACrB;AAAA,MACF;AACA,cAAQ,IAAI,iDAAiD;AAC7D,WAAK,UAAU;AACf;AAAA,IACF;AACA,SAAK,IAAI,UAAU,mBAAmB,gCAAgC;AACtE,QAAI,CAAC,KAAK,SAAS,kBAAkB;AACnC,YAAM,KAAK,IAAI,UAAU,aAAa,KAAK,EAAE,aAAa;AAAA,QACxD,MAAM;AAAA,QACN,QAAQ;AAAA,MACV,CAAC;AAAA,IACH,OAAO;AACL,YAAM,KAAK,IAAI,UAAU,QAAQ,IAAI,EAAE,aAAa;AAAA,QAClD,MAAM;AAAA,QACN,QAAQ;AAAA,MACV,CAAC;AAAA,IACH;AACA,SAAK,IAAI,UAAU;AAAA,MACjB,KAAK,IAAI,UAAU,gBAAgB,gCAAgC,EAAE,CAAC;AAAA,IACxE;AAAA,EACF;AAAA;AAAA,EAGA,MAAM,qBAAqB;AAEzB,UAAM,SAAS,MAAM,KAAK,IAAI,MAAM,SAAS,GAAG,OAAO,CAAC,SAAS,gBAAgB,SAAS,UAAU,KAAK,cAAc,QAAQ,KAAK,cAAc,SAAS;AAG3J,UAAM,aAAa,KAAK,IAAI,UAAU,gBAAgB,UAAU,EAAE,IAAI,CAAC,SAAS,KAAK,KAAK,IAAI;AAC9F,UAAM,eAAe,KAAK,eAAe,oBAAoB,KAAK;AAClE,QAAG,KAAK,SAAS,YAAW;AAC1B,WAAK,WAAW,cAAc,MAAM;AACpC,WAAK,WAAW,qBAAqB,aAAa;AAClD,WAAK,WAAW,mBAAmB,aAAa;AAAA,IAClD;AAEA,QAAI,iBAAiB,CAAC;AACtB,aAAS,IAAI,GAAG,IAAI,MAAM,QAAQ,KAAK;AAErC,UAAG,MAAM,CAAC,EAAE,KAAK,QAAQ,GAAG,IAAI,IAAI;AAElC,aAAK,cAAc,iBAAiB;AACpC;AAAA,MACF;AAEA,UAAG,KAAK,eAAe,iBAAiB,IAAI,MAAM,CAAC,EAAE,IAAI,GAAG,MAAM,CAAC,EAAE,KAAK,KAAK,GAAG;AAGhF;AAAA,MACF;AAEA,UAAG,KAAK,SAAS,aAAa,QAAQ,MAAM,CAAC,EAAE,IAAI,IAAI,IAAI;AAIzD,YAAG,KAAK,sBAAsB;AAC5B,uBAAa,KAAK,oBAAoB;AACtC,eAAK,uBAAuB;AAAA,QAC9B;AAEA,YAAG,CAAC,KAAK,4BAA2B;AAClC,cAAI,SAAS,OAAO,qFAAqF;AACzG,eAAK,6BAA6B;AAClC,qBAAW,MAAM;AACf,iBAAK,6BAA6B;AAAA,UACpC,GAAG,GAAM;AAAA,QACX;AACA;AAAA,MACF;AAEA,UAAI,OAAO;AACX,iBAAW,iBAAiB,KAAK,iBAAiB;AAChD,YAAG,MAAM,CAAC,EAAE,KAAK,SAAS,aAAa,GAAG;AACxC,iBAAO;AACP,eAAK,cAAc,aAAa;AAEhC;AAAA,QACF;AAAA,MACF;AACA,UAAG,MAAM;AACP;AAAA,MACF;AAEA,UAAG,WAAW,QAAQ,MAAM,CAAC,CAAC,IAAI,IAAI;AAEpC;AAAA,MACF;AACA,UAAI;AAEF,uBAAe,KAAK,KAAK,oBAAoB,MAAM,CAAC,GAAG,KAAK,CAAC;AAAA,MAC/D,SAAS,OAAP;AACA,gBAAQ,IAAI,KAAK;AAAA,MACnB;AAEA,UAAG,eAAe,SAAS,GAAG;AAE5B,cAAM,QAAQ,IAAI,cAAc;AAEhC,yBAAiB,CAAC;AAAA,MACpB;AAGA,UAAG,IAAI,KAAK,IAAI,QAAQ,GAAG;AACzB,cAAM,KAAK,wBAAwB;AAAA,MACrC;AAAA,IACF;AAEA,UAAM,QAAQ,IAAI,cAAc;AAEhC,UAAM,KAAK,wBAAwB;AAEnC,QAAG,KAAK,WAAW,kBAAkB,SAAS,GAAG;AAC/C,YAAM,KAAK,uBAAuB;AAAA,IACpC;AAAA,EACF;AAAA,EAEA,MAAM,wBAAwB,QAAM,OAAO;AACzC,QAAG,CAAC,KAAK,oBAAmB;AAC1B;AAAA,IACF;AAEA,QAAG,CAAC,OAAO;AAET,UAAG,KAAK,cAAc;AACpB,qBAAa,KAAK,YAAY;AAC9B,aAAK,eAAe;AAAA,MACtB;AACA,WAAK,eAAe,WAAW,MAAM;AAEnC,aAAK,wBAAwB,IAAI;AAEjC,YAAG,KAAK,cAAc;AACpB,uBAAa,KAAK,YAAY;AAC9B,eAAK,eAAe;AAAA,QACtB;AAAA,MACF,GAAG,GAAK;AACR,cAAQ,IAAI,gBAAgB;AAC5B;AAAA,IACF;AAEA,QAAG;AAED,YAAM,KAAK,eAAe,KAAK;AAC/B,WAAK,qBAAqB;AAAA,IAC5B,SAAO,OAAN;AACC,cAAQ,IAAI,KAAK;AACjB,UAAI,SAAS,OAAO,wBAAsB,MAAM,OAAO;AAAA,IACzD;AAAA,EAEF;AAAA;AAAA,EAEA,MAAM,yBAA0B;AAE9B,QAAI,oBAAoB,CAAC;AAEzB,UAAM,gCAAgC,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,0CAA0C;AACpH,QAAG,+BAA+B;AAChC,0BAAoB,MAAM,KAAK,IAAI,MAAM,QAAQ,KAAK,0CAA0C;AAEhG,0BAAoB,kBAAkB,MAAM,MAAM;AAAA,IACpD;AAEA,wBAAoB,kBAAkB,OAAO,KAAK,WAAW,iBAAiB;AAE9E,wBAAoB,CAAC,GAAG,IAAI,IAAI,iBAAiB,CAAC;AAElD,sBAAkB,KAAK;AAEvB,wBAAoB,kBAAkB,KAAK,MAAM;AAEjD,UAAM,KAAK,IAAI,MAAM,QAAQ,MAAM,4CAA4C,iBAAiB;AAEhG,UAAM,KAAK,kBAAkB;AAAA,EAC/B;AAAA;AAAA,EAGA,MAAM,oBAAqB;AAEzB,UAAM,gCAAgC,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,0CAA0C;AACpH,QAAG,CAAC,+BAA+B;AACjC,WAAK,SAAS,eAAe,CAAC;AAC9B,cAAQ,IAAI,kBAAkB;AAC9B;AAAA,IACF;AAEA,UAAM,oBAAoB,MAAM,KAAK,IAAI,MAAM,QAAQ,KAAK,0CAA0C;AAEtG,UAAM,0BAA0B,kBAAkB,MAAM,MAAM;AAE9D,UAAM,eAAe,wBAAwB,IAAI,eAAa,UAAU,MAAM,GAAG,EAAE,CAAC,CAAC,EAAE,OAAO,CAAC,QAAQ,SAAS,OAAO,SAAS,IAAI,IAAI,SAAS,CAAC,GAAG,QAAQ,IAAI,GAAG,CAAC,CAAC;AAEtK,SAAK,SAAS,eAAe;AAAA,EAE/B;AAAA;AAAA,EAEA,MAAM,qBAAsB;AAE1B,SAAK,SAAS,eAAe,CAAC;AAE9B,UAAM,gCAAgC,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,0CAA0C;AACpH,QAAG,+BAA+B;AAChC,YAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,0CAA0C;AAAA,IAChF;AAEA,UAAM,KAAK,mBAAmB;AAAA,EAChC;AAAA;AAAA,EAIA,MAAM,mBAAmB;AACvB,QAAG,CAAE,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,YAAY,GAAI;AACvD;AAAA,IACF;AACA,QAAI,iBAAiB,MAAM,KAAK,IAAI,MAAM,QAAQ,KAAK,YAAY;AAEnE,QAAI,eAAe,QAAQ,oBAAoB,IAAI,GAAG;AAEpD,UAAI,mBAAmB;AACvB,0BAAoB;AACpB,YAAM,KAAK,IAAI,MAAM,QAAQ,MAAM,cAAc,iBAAiB,gBAAgB;AAClF,cAAQ,IAAI,wCAAwC;AAAA,IACtD;AAAA,EACF;AAAA;AAAA,EAGA,MAAM,gCAAgC;AACpC,QAAI,SAAS,OAAO,+EAA+E;AAEnG,UAAM,KAAK,eAAe,cAAc;AAExC,UAAM,KAAK,mBAAmB;AAC9B,SAAK,kBAAkB;AACvB,QAAI,SAAS,OAAO,2EAA2E;AAAA,EACjG;AAAA;AAAA,EAGA,MAAM,oBAAoB,WAAW,OAAK,MAAM;AAE9C,QAAI,YAAY,CAAC;AACjB,QAAI,SAAS,CAAC;AAEd,UAAM,gBAAgB,IAAI,UAAU,IAAI;AAExC,QAAI,mBAAmB,UAAU,KAAK,QAAQ,OAAO,EAAE;AACvD,uBAAmB,iBAAiB,QAAQ,OAAO,KAAK;AAExD,QAAI,YAAY;AAChB,aAAQ,IAAI,GAAG,IAAI,KAAK,UAAU,QAAQ,KAAK;AAC7C,UAAG,UAAU,KAAK,QAAQ,KAAK,UAAU,CAAC,CAAC,IAAI,IAAI;AACjD,oBAAY;AACZ,gBAAQ,IAAI,mCAAmC,KAAK,UAAU,CAAC,CAAC;AAEhE;AAAA,MACF;AAAA,IACF;AAEA,QAAG,WAAW;AACZ,gBAAU,KAAK,CAAC,eAAe,kBAAkB;AAAA,QAC/C,OAAO,UAAU,KAAK;AAAA,QACtB,MAAM,UAAU;AAAA,MAClB,CAAC,CAAC;AACF,YAAM,KAAK,qBAAqB,SAAS;AACzC;AAAA,IACF;AAIA,QAAG,UAAU,cAAc,UAAU;AAEnC,YAAM,kBAAkB,MAAM,KAAK,IAAI,MAAM,WAAW,SAAS;AACjE,UAAI,OAAO,oBAAoB,YAAc,gBAAgB,QAAQ,OAAO,IAAI,IAAK;AACnF,cAAM,cAAc,KAAK,MAAM,eAAe;AAE9C,iBAAQ,IAAI,GAAG,IAAI,YAAY,MAAM,QAAQ,KAAK;AAEhD,cAAG,YAAY,MAAM,CAAC,EAAE,MAAM;AAE5B,gCAAoB,OAAO,YAAY,MAAM,CAAC,EAAE;AAAA,UAClD;AAEA,cAAG,YAAY,MAAM,CAAC,EAAE,MAAM;AAE5B,gCAAoB,aAAa,YAAY,MAAM,CAAC,EAAE;AAAA,UACxD;AAAA,QACF;AAAA,MACF;AAEA,gBAAU,KAAK,CAAC,eAAe,kBAAkB;AAAA,QAC/C,OAAO,UAAU,KAAK;AAAA,QACtB,MAAM,UAAU;AAAA,MAClB,CAAC,CAAC;AACF,YAAM,KAAK,qBAAqB,SAAS;AACzC;AAAA,IACF;AAMA,UAAM,gBAAgB,MAAM,KAAK,IAAI,MAAM,WAAW,SAAS;AAC/D,QAAI,4BAA4B;AAChC,UAAM,gBAAgB,KAAK,aAAa,eAAe,UAAU,IAAI;AAGrE,QAAG,cAAc,SAAS,GAAG;AAG3B,eAAS,IAAI,GAAG,IAAI,cAAc,QAAQ,KAAK;AAE7C,cAAM,oBAAoB,cAAc,CAAC,EAAE;AAG3C,cAAM,YAAY,IAAI,cAAc,CAAC,EAAE,IAAI;AAC3C,eAAO,KAAK,SAAS;AAGrB,YAAI,KAAK,eAAe,SAAS,SAAS,MAAM,kBAAkB,QAAQ;AAGxE;AAAA,QACF;AAGA,YAAG,KAAK,eAAe,iBAAiB,WAAW,UAAU,KAAK,KAAK,GAAG;AAGxE;AAAA,QACF;AAEA,cAAM,aAAa,IAAI,kBAAkB,KAAK,CAAC;AAC/C,YAAG,KAAK,eAAe,SAAS,SAAS,MAAM,YAAY;AAGzD;AAAA,QACF;AAGA,kBAAU,KAAK,CAAC,WAAW,mBAAmB;AAAA;AAAA;AAAA,UAG5C,OAAO,KAAK,IAAI;AAAA,UAChB,MAAM;AAAA,UACN,QAAQ;AAAA,UACR,MAAM,cAAc,CAAC,EAAE;AAAA,UACvB,MAAM,kBAAkB;AAAA,QAC1B,CAAC,CAAC;AACF,YAAG,UAAU,SAAS,GAAG;AAEvB,gBAAM,KAAK,qBAAqB,SAAS;AACzC,uCAA6B,UAAU;AAGvC,cAAI,6BAA6B,IAAI;AAEnC,kBAAM,KAAK,wBAAwB;AAEnC,wCAA4B;AAAA,UAC9B;AAEA,sBAAY,CAAC;AAAA,QACf;AAAA,MACF;AAAA,IACF;AAEA,QAAG,UAAU,SAAS,GAAG;AAEvB,YAAM,KAAK,qBAAqB,SAAS;AACzC,kBAAY,CAAC;AACb,mCAA6B,UAAU;AAAA,IACzC;AAQA,wBAAoB;AAAA;AAIpB,QAAG,cAAc,SAAS,yBAAyB;AACjD,0BAAoB;AAAA,IACtB,OAAK;AACH,YAAM,kBAAkB,KAAK,IAAI,cAAc,aAAa,SAAS;AAErE,UAAG,OAAO,gBAAgB,aAAa,aAAa;AAElD,4BAAoB,cAAc,UAAU,GAAG,uBAAuB;AAAA,MACxE,OAAK;AACH,YAAI,gBAAgB;AACpB,iBAAS,IAAI,GAAG,IAAI,gBAAgB,SAAS,QAAQ,KAAK;AAExD,gBAAM,gBAAgB,gBAAgB,SAAS,CAAC,EAAE;AAElD,gBAAM,eAAe,gBAAgB,SAAS,CAAC,EAAE;AAEjD,cAAI,aAAa;AACjB,mBAAS,IAAI,GAAG,IAAI,eAAe,KAAK;AACtC,0BAAc;AAAA,UAChB;AAEA,2BAAiB,GAAG,cAAc;AAAA;AAAA,QACpC;AAEA,4BAAoB;AACpB,YAAG,iBAAiB,SAAS,yBAAyB;AACpD,6BAAmB,iBAAiB,UAAU,GAAG,uBAAuB;AAAA,QAC1E;AAAA,MACF;AAAA,IACF;AAGA,UAAM,YAAY,IAAI,iBAAiB,KAAK,CAAC;AAC7C,UAAM,gBAAgB,KAAK,eAAe,SAAS,aAAa;AAChE,QAAG,iBAAkB,cAAc,eAAgB;AAEjD,WAAK,kBAAkB,QAAQ,gBAAgB;AAC/C;AAAA,IACF;AAAC;AAGD,UAAM,kBAAkB,KAAK,eAAe,aAAa,aAAa;AACtE,QAAI,0BAA0B;AAC9B,QAAG,mBAAmB,MAAM,QAAQ,eAAe,KAAM,OAAO,SAAS,GAAI;AAE3E,eAAS,IAAI,GAAG,IAAI,OAAO,QAAQ,KAAK;AACtC,YAAG,gBAAgB,QAAQ,OAAO,CAAC,CAAC,MAAM,IAAI;AAC5C,oCAA0B;AAC1B;AAAA,QACF;AAAA,MACF;AAAA,IACF;AAEA,QAAG,yBAAwB;AAEzB,YAAM,iBAAiB,UAAU,KAAK;AAEtC,YAAM,iBAAiB,KAAK,eAAe,SAAS,aAAa;AACjE,UAAI,gBAAgB;AAElB,cAAM,iBAAiB,KAAK,MAAO,KAAK,IAAI,iBAAiB,cAAc,IAAI,iBAAkB,GAAG;AACpG,YAAG,iBAAiB,IAAI;AAGtB,eAAK,WAAW,kBAAkB,UAAU,IAAI,IAAI,iBAAiB;AACrE,eAAK,kBAAkB,QAAQ,gBAAgB;AAC/C;AAAA,QACF;AAAA,MACF;AAAA,IACF;AACA,QAAI,OAAO;AAAA,MACT,OAAO,UAAU,KAAK;AAAA,MACtB,MAAM;AAAA,MACN,MAAM,UAAU;AAAA,MAChB,MAAM,UAAU,KAAK;AAAA,MACrB,UAAU;AAAA,IACZ;AAEA,cAAU,KAAK,CAAC,eAAe,kBAAkB,IAAI,CAAC;AAEtD,UAAM,KAAK,qBAAqB,SAAS;AAIzC,QAAI,MAAM;AAER,YAAM,KAAK,wBAAwB;AAAA,IACrC;AAAA,EAEF;AAAA,EAEA,kBAAkB,QAAQ,kBAAkB;AAC1C,QAAI,OAAO,SAAS,GAAG;AAErB,WAAK,WAAW,yBAAyB,iBAAiB,SAAS;AAAA,IACrE,OAAO;AAEL,WAAK,WAAW,yBAAyB,iBAAiB,SAAS;AAAA,IACrE;AAAA,EACF;AAAA,EAEA,MAAM,qBAAqB,WAAW;AACpC,YAAQ,IAAI,sBAAsB;AAElC,QAAG,UAAU,WAAW;AAAG;AAE3B,UAAM,eAAe,UAAU,IAAI,CAAC,QAAQ,IAAI,CAAC,CAAC;AAElD,UAAM,iBAAiB,MAAM,KAAK,6BAA6B,YAAY;AAE3E,QAAG,CAAC,gBAAgB;AAClB,cAAQ,IAAI,wBAAwB;AAEpC,WAAK,WAAW,oBAAoB,CAAC,GAAG,KAAK,WAAW,mBAAmB,GAAG,UAAU,IAAI,CAAC,QAAQ,IAAI,CAAC,EAAE,IAAI,CAAC;AACjH;AAAA,IACF;AAEA,QAAG,gBAAe;AAChB,WAAK,qBAAqB;AAE1B,UAAG,KAAK,SAAS,YAAW;AAC1B,YAAG,KAAK,SAAS,kBAAiB;AAChC,eAAK,WAAW,QAAQ,CAAC,GAAG,KAAK,WAAW,OAAO,GAAG,UAAU,IAAI,CAAC,QAAQ,IAAI,CAAC,EAAE,IAAI,CAAC;AAAA,QAC3F;AACA,aAAK,WAAW,kBAAkB,UAAU;AAE5C,aAAK,WAAW,eAAe,eAAe,MAAM;AAAA,MACtD;AAGA,eAAQ,IAAI,GAAG,IAAI,eAAe,KAAK,QAAQ,KAAK;AAClD,cAAM,MAAM,eAAe,KAAK,CAAC,EAAE;AACnC,cAAM,QAAQ,eAAe,KAAK,CAAC,EAAE;AACrC,YAAG,KAAK;AACN,gBAAM,MAAM,UAAU,KAAK,EAAE,CAAC;AAC9B,gBAAM,OAAO,UAAU,KAAK,EAAE,CAAC;AAC/B,eAAK,eAAe,eAAe,KAAK,KAAK,IAAI;AAAA,QACnD;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAAA,EAEA,MAAM,6BAA6B,aAAa,UAAU,GAAG;AAS3D,QAAG,YAAY,WAAW,GAAG;AAC3B,cAAQ,IAAI,sBAAsB;AAClC,aAAO;AAAA,IACT;AACA,UAAM,aAAa;AAAA,MACjB,OAAO;AAAA,MACP,OAAO;AAAA,IACT;AAEA,UAAM,YAAY;AAAA,MAChB,KAAK;AAAA,MACL,QAAQ;AAAA,MACR,MAAM,KAAK,UAAU,UAAU;AAAA,MAC/B,SAAS;AAAA,QACP,gBAAgB;AAAA,QAChB,iBAAiB,UAAU,KAAK,SAAS;AAAA,MAC3C;AAAA,IACF;AACA,QAAI;AACJ,QAAI;AACF,aAAO,OAAO,GAAG,SAAS,SAAS,SAAS;AAC5C,aAAO,KAAK,MAAM,IAAI;AAAA,IACxB,SAAS,OAAP;AAEA,UAAI,MAAM,WAAW,OAAS,UAAU,GAAI;AAC1C;AAEA,cAAM,UAAU,KAAK,IAAI,SAAS,CAAC;AACnC,gBAAQ,IAAI,6BAA6B,oBAAoB;AAC7D,cAAM,IAAI,QAAQ,OAAK,WAAW,GAAG,MAAO,OAAO,CAAC;AACpD,eAAO,MAAM,KAAK,6BAA6B,aAAa,OAAO;AAAA,MACrE;AAEA,cAAQ,IAAI,IAAI;AAOhB,cAAQ,IAAI,KAAK;AAGjB,aAAO;AAAA,IACT;AAAA,EACF;AAAA,EACA,MAAM,eAAe;AACnB,UAAM,cAAc;AACpB,UAAM,OAAO,MAAM,KAAK,6BAA6B,WAAW;AAChE,QAAG,QAAQ,KAAK,OAAO;AACrB,cAAQ,IAAI,kBAAkB;AAC9B,aAAO;AAAA,IACT,OAAK;AACH,cAAQ,IAAI,oBAAoB;AAChC,aAAO;AAAA,IACT;AAAA,EACF;AAAA,EAGA,oBAAoB;AAElB,QAAG,KAAK,SAAS,YAAY;AAC3B,UAAI,KAAK,WAAW,mBAAmB,GAAG;AACxC;AAAA,MACF,OAAK;AAEH,gBAAQ,IAAI,KAAK,UAAU,KAAK,YAAY,MAAM,CAAC,CAAC;AAAA,MACtD;AAAA,IACF;AAGA,SAAK,aAAa,CAAC;AACnB,SAAK,WAAW,qBAAqB;AACrC,SAAK,WAAW,kBAAkB,CAAC;AACnC,SAAK,WAAW,oBAAoB,CAAC;AACrC,SAAK,WAAW,QAAQ,CAAC;AACzB,SAAK,WAAW,iBAAiB;AACjC,SAAK,WAAW,oBAAoB,CAAC;AACrC,SAAK,WAAW,cAAc;AAC9B,SAAK,WAAW,wBAAwB;AAAA,EAC1C;AAAA;AAAA,EAGA,MAAM,sBAAsB,eAAa,MAAM;AAE7C,UAAM,WAAW,IAAI,aAAa,IAAI;AAGtC,QAAI,UAAU,CAAC;AACf,QAAG,KAAK,cAAc,QAAQ,GAAG;AAC/B,gBAAU,KAAK,cAAc,QAAQ;AAAA,IAEvC,OAAK;AAEH,eAAQ,IAAI,GAAG,IAAI,KAAK,gBAAgB,QAAQ,KAAK;AACnD,YAAG,aAAa,KAAK,QAAQ,KAAK,gBAAgB,CAAC,CAAC,IAAI,IAAI;AAC1D,eAAK,cAAc,KAAK,gBAAgB,CAAC,CAAC;AAE1C,iBAAO;AAAA,QACT;AAAA,MACF;AAIA,iBAAW,MAAM;AACf,aAAK,mBAAmB;AAAA,MAC1B,GAAG,GAAI;AAEP,UAAG,KAAK,eAAe,iBAAiB,UAAU,aAAa,KAAK,KAAK,GAAG;AAAA,MAG5E,OAAK;AAEH,cAAM,KAAK,oBAAoB,YAAY;AAAA,MAC7C;AAEA,YAAM,MAAM,KAAK,eAAe,QAAQ,QAAQ;AAChD,UAAG,CAAC,KAAK;AACP,eAAO,mCAAiC,aAAa;AAAA,MACvD;AAGA,gBAAU,KAAK,eAAe,QAAQ,KAAK;AAAA,QACzC,UAAU;AAAA,QACV,eAAe,KAAK,SAAS;AAAA,MAC/B,CAAC;AAGD,WAAK,cAAc,QAAQ,IAAI;AAAA,IACjC;AAGA,WAAO;AAAA,EACT;AAAA;AAAA,EAGA,cAAc,WAAW;AAEvB,SAAK,WAAW,gBAAgB,SAAS,KAAK,KAAK,WAAW,gBAAgB,SAAS,KAAK,KAAK;AAAA,EACnG;AAAA,EAGA,aAAa,UAAU,WAAU;AAE/B,QAAG,KAAK,SAAS,eAAe;AAC9B,aAAO,CAAC;AAAA,IACV;AAEA,UAAM,QAAQ,SAAS,MAAM,IAAI;AAEjC,QAAI,SAAS,CAAC;AAEd,QAAI,iBAAiB,CAAC;AAEtB,UAAM,mBAAmB,UAAU,QAAQ,OAAO,EAAE,EAAE,QAAQ,OAAO,KAAK;AAE1E,QAAI,QAAQ;AACZ,QAAI,iBAAiB;AACrB,QAAI,aAAa;AAEjB,QAAI,oBAAoB;AACxB,QAAI,IAAI;AACR,QAAI,sBAAsB,CAAC;AAE3B,SAAK,IAAI,GAAG,IAAI,MAAM,QAAQ,KAAK;AAEjC,YAAM,OAAO,MAAM,CAAC;AAIpB,UAAI,CAAC,KAAK,WAAW,GAAG,KAAM,CAAC,KAAI,GAAG,EAAE,QAAQ,KAAK,CAAC,CAAC,IAAI,GAAG;AAE5D,YAAG,SAAS;AAAI;AAEhB,YAAG,CAAC,MAAM,QAAQ,EAAE,QAAQ,IAAI,IAAI;AAAI;AAExC,YAAG,eAAe,WAAW;AAAG;AAEhC,iBAAS,OAAO;AAChB;AAAA,MACF;AAKA,0BAAoB;AAEpB,UAAG,IAAI,KAAM,sBAAuB,IAAE,KAAQ,MAAM,QAAQ,IAAI,IAAI,MAAO,KAAK,kBAAkB,cAAc,GAAG;AACjH,qBAAa;AAAA,MACf;AAEA,YAAM,QAAQ,KAAK,MAAM,GAAG,EAAE,SAAS;AAEvC,uBAAiB,eAAe,OAAO,YAAU,OAAO,QAAQ,KAAK;AAGrE,qBAAe,KAAK,EAAC,QAAQ,KAAK,QAAQ,MAAM,EAAE,EAAE,KAAK,GAAG,MAAY,CAAC;AAEzE,cAAQ;AACR,eAAS,OAAO,eAAe,IAAI,YAAU,OAAO,MAAM,EAAE,KAAK,KAAK;AACtE,uBAAiB,MAAI,eAAe,IAAI,YAAU,OAAO,MAAM,EAAE,KAAK,GAAG;AAEzE,UAAG,oBAAoB,QAAQ,cAAc,IAAI,IAAI;AACnD,YAAI,QAAQ;AACZ,eAAM,oBAAoB,QAAQ,GAAG,kBAAkB,QAAQ,IAAI,IAAI;AACrE;AAAA,QACF;AACA,yBAAiB,GAAG,kBAAkB;AAAA,MACxC;AACA,0BAAoB,KAAK,cAAc;AACvC,mBAAa,YAAY;AAAA,IAC3B;AAEA,QAAI,sBAAuB,IAAE,KAAQ,MAAM,QAAQ,IAAI,IAAI,MAAO,KAAK,kBAAkB,cAAc;AAAG,mBAAa;AAEvH,aAAS,OAAO,OAAO,OAAK,EAAE,SAAS,EAAE;AAGzC,WAAO;AAEP,aAAS,eAAe;AAEtB,YAAM,qBAAqB,MAAM,QAAQ,IAAI,IAAI;AACjD,YAAM,eAAe,MAAM,SAAS;AAEpC,UAAI,MAAM,SAAS,yBAAyB;AAC1C,gBAAQ,MAAM,UAAU,GAAG,uBAAuB;AAAA,MACpD;AACA,aAAO,KAAK,EAAE,MAAM,MAAM,KAAK,GAAG,MAAM,YAAY,QAAQ,aAAa,CAAC;AAAA,IAC5E;AAAA,EACF;AAAA;AAAA,EAEA,MAAM,gBAAgB,MAAM,SAAO,CAAC,GAAG;AACrC,aAAS;AAAA,MACP,OAAO;AAAA,MACP,gBAAgB;AAAA,MAChB,WAAW;AAAA,MACX,GAAG;AAAA,IACL;AAEA,QAAI,KAAK,QAAQ,GAAG,IAAI,GAAG;AACzB,cAAQ,IAAI,uBAAqB,IAAI;AACrC,aAAO;AAAA,IACT;AACA,QAAI,QAAQ,CAAC;AACb,QAAI,iBAAiB,KAAK,MAAM,GAAG,EAAE,MAAM,CAAC;AAE5C,QAAI,qBAAqB;AACzB,QAAG,eAAe,eAAe,SAAO,CAAC,EAAE,QAAQ,GAAG,IAAI,IAAI;AAE5D,2BAAqB,SAAS,eAAe,eAAe,SAAO,CAAC,EAAE,MAAM,GAAG,EAAE,CAAC,EAAE,QAAQ,KAAK,EAAE,CAAC;AAEpG,qBAAe,eAAe,SAAO,CAAC,IAAI,eAAe,eAAe,SAAO,CAAC,EAAE,MAAM,GAAG,EAAE,CAAC;AAAA,IAChG;AACA,QAAI,iBAAiB,CAAC;AACtB,QAAI,mBAAmB;AACvB,QAAI,aAAa;AACjB,QAAI,IAAI;AAER,UAAM,YAAY,KAAK,MAAM,GAAG,EAAE,CAAC;AAEnC,UAAM,OAAO,KAAK,IAAI,MAAM,sBAAsB,SAAS;AAC3D,QAAG,EAAE,gBAAgB,SAAS,QAAQ;AACpC,cAAQ,IAAI,iBAAe,SAAS;AACpC,aAAO;AAAA,IACT;AAEA,UAAM,gBAAgB,MAAM,KAAK,IAAI,MAAM,WAAW,IAAI;AAE1D,UAAM,QAAQ,cAAc,MAAM,IAAI;AAEtC,QAAI,UAAU;AACd,SAAK,IAAI,GAAG,IAAI,MAAM,QAAQ,KAAK;AAEjC,YAAM,OAAO,MAAM,CAAC;AAEpB,UAAG,KAAK,QAAQ,KAAK,MAAM,GAAG;AAC5B,kBAAU,CAAC;AAAA,MACb;AAEA,UAAG,SAAS;AACV;AAAA,MACF;AAEA,UAAG,CAAC,MAAM,QAAQ,EAAE,QAAQ,IAAI,IAAI;AAAI;AAIxC,UAAI,CAAC,KAAK,WAAW,GAAG,KAAM,CAAC,KAAI,GAAG,EAAE,QAAQ,KAAK,CAAC,CAAC,IAAI,GAAG;AAC5D;AAAA,MACF;AAMA,YAAM,eAAe,KAAK,QAAQ,MAAM,EAAE,EAAE,KAAK;AAEjD,YAAM,gBAAgB,eAAe,QAAQ,YAAY;AACzD,UAAI,gBAAgB;AAAG;AAEvB,UAAI,eAAe,WAAW;AAAe;AAE7C,qBAAe,KAAK,YAAY;AAEhC,UAAI,eAAe,WAAW,eAAe,QAAQ;AAEnD,YAAG,uBAAuB,GAAG;AAE3B,uBAAa,IAAI;AACjB;AAAA,QACF;AAEA,YAAG,qBAAqB,oBAAmB;AACzC,uBAAa,IAAI;AACjB;AAAA,QACF;AACA;AAEA,uBAAe,IAAI;AACnB;AAAA,MACF;AAAA,IACF;AAEA,QAAI,eAAe;AAAG,aAAO;AAE7B,cAAU;AAEV,QAAI,aAAa;AACjB,SAAK,IAAI,YAAY,IAAI,MAAM,QAAQ,KAAK;AAC1C,UAAI,OAAO,eAAe,YAAc,MAAM,SAAS,YAAY;AACjE,cAAM,KAAK,KAAK;AAChB;AAAA,MACF;AACA,UAAI,OAAO,MAAM,CAAC;AAClB,UAAK,KAAK,QAAQ,GAAG,MAAM,KAAO,CAAC,KAAI,GAAG,EAAE,QAAQ,KAAK,CAAC,CAAC,MAAM,IAAI;AACnE;AAAA,MACF;AAGA,UAAI,OAAO,aAAa,aAAa,OAAO,WAAW;AACrD,cAAM,KAAK,KAAK;AAChB;AAAA,MACF;AAEA,UAAI,OAAO,aAAe,KAAK,SAAS,aAAc,OAAO,WAAY;AACvE,cAAM,gBAAgB,OAAO,YAAY;AACzC,eAAO,KAAK,MAAM,GAAG,aAAa,IAAI;AACtC;AAAA,MACF;AAGA,UAAI,KAAK,WAAW;AAAG;AAEvB,UAAI,OAAO,kBAAkB,KAAK,SAAS,OAAO,gBAAgB;AAChE,eAAO,KAAK,MAAM,GAAG,OAAO,cAAc,IAAI;AAAA,MAChD;AAEA,UAAI,KAAK,WAAW,KAAK,GAAG;AAC1B,kBAAU,CAAC;AACX;AAAA,MACF;AACA,UAAI,SAAQ;AAEV,eAAO,MAAK;AAAA,MACd;AAEA,YAAM,KAAK,IAAI;AAEf,oBAAc,KAAK;AAAA,IACrB;AAEA,QAAI,SAAS;AACX,YAAM,KAAK,KAAK;AAAA,IAClB;AACA,WAAO,MAAM,KAAK,IAAI,EAAE,KAAK;AAAA,EAC/B;AAAA;AAAA,EAGA,MAAM,eAAe,MAAM,SAAO,CAAC,GAAG;AACpC,aAAS;AAAA,MACP,OAAO;AAAA,MACP,WAAW;AAAA,MACX,gBAAgB;AAAA,MAChB,GAAG;AAAA,IACL;AACA,UAAM,YAAY,KAAK,IAAI,MAAM,sBAAsB,IAAI;AAE3D,QAAI,EAAE,qBAAqB,SAAS;AAAgB,aAAO;AAE3D,UAAM,eAAe,MAAM,KAAK,IAAI,MAAM,WAAW,SAAS;AAC9D,UAAM,aAAa,aAAa,MAAM,IAAI;AAC1C,QAAI,kBAAkB,CAAC;AACvB,QAAI,UAAU;AACd,QAAI,aAAa;AACjB,UAAMA,cAAa,OAAO,SAAS,WAAW;AAC9C,aAAS,IAAI,GAAG,gBAAgB,SAASA,aAAY,KAAK;AACxD,UAAI,OAAO,WAAW,CAAC;AAEvB,UAAI,OAAO,SAAS;AAClB;AAEF,UAAI,KAAK,WAAW;AAClB;AAEF,UAAI,OAAO,kBAAkB,KAAK,SAAS,OAAO,gBAAgB;AAChE,eAAO,KAAK,MAAM,GAAG,OAAO,cAAc,IAAI;AAAA,MAChD;AAEA,UAAI,SAAS;AACX;AAEF,UAAI,CAAC,MAAM,QAAQ,EAAE,QAAQ,IAAI,IAAI;AACnC;AAEF,UAAI,KAAK,QAAQ,KAAK,MAAM,GAAG;AAC7B,kBAAU,CAAC;AACX;AAAA,MACF;AAEA,UAAI,OAAO,aAAa,aAAa,OAAO,WAAW;AACrD,wBAAgB,KAAK,KAAK;AAC1B;AAAA,MACF;AACA,UAAI,SAAS;AAEX,eAAO,MAAO;AAAA,MAChB;AAEA,UAAI,gBAAgB,IAAI,GAAG;AAIzB,YAAK,gBAAgB,SAAS,KAAM,gBAAgB,gBAAgB,gBAAgB,SAAS,CAAC,CAAC,GAAG;AAEhG,0BAAgB,IAAI;AAAA,QACtB;AAAA,MACF;AAEA,sBAAgB,KAAK,IAAI;AAEzB,oBAAc,KAAK;AAAA,IACrB;AAEA,aAAS,IAAI,GAAG,IAAI,gBAAgB,QAAQ,KAAK;AAE/C,UAAI,gBAAgB,gBAAgB,CAAC,CAAC,GAAG;AAEvC,YAAI,MAAM,gBAAgB,SAAS,GAAG;AAEpC,0BAAgB,IAAI;AACpB;AAAA,QACF;AAEA,wBAAgB,CAAC,IAAI,gBAAgB,CAAC,EAAE,QAAQ,MAAM,EAAE;AACxD,wBAAgB,CAAC,IAAI;AAAA,EAAK,gBAAgB,CAAC;AAAA,MAC7C;AAAA,IACF;AAEA,sBAAkB,gBAAgB,KAAK,IAAI;AAC3C,WAAO;AAAA,EACT;AAAA;AAAA,EAGA,kBAAkB,gBAAgB;AAChC,QAAI,QAAQ;AACZ,QAAI,KAAK,kBAAkB,SAAS,GAAG;AACrC,eAAS,IAAI,GAAG,IAAI,KAAK,kBAAkB,QAAQ,KAAK;AACtD,YAAI,eAAe,QAAQ,KAAK,kBAAkB,CAAC,CAAC,IAAI,IAAI;AAC1D,kBAAQ;AACR,eAAK,cAAc,cAAY,KAAK,kBAAkB,CAAC,CAAC;AACxD;AAAA,QACF;AAAA,MACF;AAAA,IACF;AACA,WAAO;AAAA,EACT;AAAA;AAAA,EAEA,aAAa,WAAW,WAAS,WAAW;AAE1C,QAAI,cAAc,OAAO;AACvB,YAAM,YAAY,OAAO,KAAK,KAAK,WAAW;AAC9C,eAAS,IAAI,GAAG,IAAI,UAAU,QAAQ,KAAK;AACzC,aAAK,aAAa,KAAK,YAAY,UAAU,CAAC,CAAC,GAAG,UAAU,CAAC,CAAC;AAAA,MAChE;AACA;AAAA,IACF;AAEA,SAAK,YAAY,QAAQ,IAAI;AAE7B,QAAI,KAAK,YAAY,QAAQ,EAAE,cAAc,WAAW,GAAG;AACzD,WAAK,YAAY,QAAQ,EAAE,cAAc,WAAW,EAAE,OAAO;AAAA,IAC/D;AACA,UAAM,kBAAkB,KAAK,YAAY,QAAQ,EAAE,SAAS,OAAO,EAAE,KAAK,WAAW,CAAC;AAGtF,aAAS,QAAQ,iBAAiB,mBAAmB;AACrD,UAAM,UAAU,gBAAgB,SAAS,GAAG;AAC5C,QAAI,OAAO;AACX,QAAI,OAAO,CAAC;AAEZ,QAAI,KAAK,kBAAkB;AACzB,aAAO;AACP,aAAO;AAAA,QACL,OAAO;AAAA,MACT;AAAA,IACF;AACA,YAAQ,SAAS,KAAK;AAAA,MACpB,KAAK;AAAA,MACL;AAAA,MACA,MAAM;AAAA,MACN,QAAQ;AAAA,MACR;AAAA,IACF,CAAC;AAAA,EACH;AAAA;AAAA,EAIA,MAAM,eAAe,WAAW,SAAS;AACvC,QAAI;AAEJ,QAAI,UAAU,SAAS,SAAS,KAAO,UAAU,SAAS,CAAC,EAAE,UAAU,SAAS,SAAS,GAAG;AAC1F,aAAO,UAAU,SAAS,CAAC;AAAA,IAC7B;AAEA,QAAI,MAAM;AACR,WAAK,MAAM;AAAA,IACb,OAAO;AAEL,aAAO,UAAU,SAAS,OAAO,EAAE,KAAK,UAAU,CAAC;AAAA,IACrD;AACA,QAAI,sBAAsB;AAE1B,QAAG,CAAC,KAAK,SAAS;AAAe,6BAAuB;AAGxD,QAAG,CAAC,KAAK,SAAS,uBAAuB;AAEvC,eAAS,IAAI,GAAG,IAAI,QAAQ,QAAQ,KAAK;AAKvC,gBAAQ,IAAI,IAAI;AAChB,YAAI,OAAO,QAAQ,CAAC,EAAE,SAAS,UAAU;AACvC,gBAAMC,QAAO,KAAK,SAAS,OAAO,EAAE,KAAK,gBAAgB,CAAC;AAC1D,gBAAMC,QAAOD,MAAK,SAAS,KAAK;AAAA,YAC9B,KAAK;AAAA,YACL,MAAM,QAAQ,CAAC,EAAE,KAAK;AAAA,YACtB,OAAO,QAAQ,CAAC,EAAE,KAAK;AAAA,UACzB,CAAC;AACD,UAAAC,MAAK,YAAY,KAAK,yBAAyB,QAAQ,CAAC,EAAE,IAAI;AAC9D,UAAAD,MAAK,QAAQ,aAAa,MAAM;AAChC;AAAA,QACF;AAKA,YAAI;AACJ,cAAM,sBAAsB,KAAK,MAAM,QAAQ,CAAC,EAAE,aAAa,GAAG,IAAI;AACtE,YAAG,KAAK,SAAS,gBAAgB;AAC/B,gBAAM,MAAM,QAAQ,CAAC,EAAE,KAAK,MAAM,GAAG;AACrC,2BAAiB,IAAI,IAAI,SAAS,CAAC;AACnC,gBAAM,OAAO,IAAI,MAAM,GAAG,IAAI,SAAS,CAAC,EAAE,KAAK,GAAG;AAElD,2BAAiB,UAAU,yBAAyB,UAAU;AAAA,QAChE,OAAK;AACH,2BAAiB,YAAY,sBAAsB,QAAQ,QAAQ,CAAC,EAAE,KAAK,MAAM,GAAG,EAAE,IAAI,IAAI;AAAA,QAChG;AAGA,YAAG,CAAC,KAAK,qBAAqB,QAAQ,CAAC,EAAE,IAAI,GAAE;AAC7C,gBAAMA,QAAO,KAAK,SAAS,OAAO,EAAE,KAAK,gBAAgB,CAAC;AAC1D,gBAAMC,QAAOD,MAAK,SAAS,KAAK;AAAA,YAC9B,KAAK;AAAA,YACL,MAAM,QAAQ,CAAC,EAAE;AAAA,UACnB,CAAC;AACD,UAAAC,MAAK,YAAY;AAEjB,UAAAD,MAAK,QAAQ,aAAa,MAAM;AAEhC,eAAK,mBAAmBC,OAAM,QAAQ,CAAC,GAAGD,KAAI;AAC9C;AAAA,QACF;AAGA,yBAAiB,eAAe,QAAQ,OAAO,EAAE,EAAE,QAAQ,MAAM,KAAK;AAEtE,cAAM,OAAO,KAAK,SAAS,OAAO,EAAE,KAAK,oBAAoB,CAAC;AAE9D,cAAM,SAAS,KAAK,SAAS,QAAQ,EAAE,KAAK,eAAe,CAAC;AAE5D,iBAAS,QAAQ,QAAQ,gBAAgB;AACzC,cAAM,OAAO,OAAO,SAAS,KAAK;AAAA,UAChC,KAAK;AAAA,UACL,OAAO,QAAQ,CAAC,EAAE;AAAA,QACpB,CAAC;AACD,aAAK,YAAY;AAEjB,aAAK,mBAAmB,MAAM,QAAQ,CAAC,GAAG,IAAI;AAC9C,eAAO,iBAAiB,SAAS,CAAC,UAAU;AAE1C,cAAI,SAAS,MAAM,OAAO;AAC1B,iBAAO,CAAC,OAAO,UAAU,SAAS,eAAe,GAAG;AAClD,qBAAS,OAAO;AAAA,UAClB;AAEA,iBAAO,UAAU,OAAO,cAAc;AAAA,QACxC,CAAC;AACD,cAAM,WAAW,KAAK,SAAS,MAAM,EAAE,KAAK,GAAG,CAAC;AAChD,cAAM,qBAAqB,SAAS,SAAS,MAAM;AAAA,UACjD,KAAK;AAAA,UACL,OAAO,QAAQ,CAAC,EAAE;AAAA,QACpB,CAAC;AACD,YAAG,QAAQ,CAAC,EAAE,KAAK,QAAQ,GAAG,IAAI,IAAG;AACnC,mBAAS,iBAAiB,eAAgB,MAAM,KAAK,gBAAgB,QAAQ,CAAC,EAAE,MAAM,EAAC,OAAO,IAAI,WAAW,IAAI,CAAC,GAAI,oBAAoB,QAAQ,CAAC,EAAE,MAAM,IAAI,SAAS,UAAU,CAAC;AAAA,QACrL,OAAK;AACH,gBAAM,kBAAkB,MAAM,KAAK,eAAe,QAAQ,CAAC,EAAE,MAAM,EAAC,OAAO,IAAI,WAAW,IAAI,CAAC;AAC/F,cAAG,CAAC;AAAiB;AACrB,mBAAS,iBAAiB,eAAe,iBAAiB,oBAAoB,QAAQ,CAAC,EAAE,MAAM,IAAI,SAAS,UAAU,CAAC;AAAA,QACzH;AACA,aAAK,mBAAmB,UAAU,QAAQ,CAAC,GAAG,IAAI;AAAA,MACpD;AACA,WAAK,aAAa,WAAW,OAAO;AACpC;AAAA,IACF;AAGA,UAAM,kBAAkB,CAAC;AACzB,aAAS,IAAI,GAAG,IAAI,QAAQ,QAAQ,KAAK;AACvC,YAAM,OAAO,QAAQ,CAAC;AACtB,YAAM,OAAO,KAAK;AAElB,UAAI,OAAO,SAAS,UAAU;AAC5B,wBAAgB,KAAK,IAAI,IAAI,CAAC,IAAI;AAClC;AAAA,MACF;AACA,UAAI,KAAK,QAAQ,GAAG,IAAI,IAAI;AAC1B,cAAM,YAAY,KAAK,MAAM,GAAG,EAAE,CAAC;AACnC,YAAI,CAAC,gBAAgB,SAAS,GAAG;AAC/B,0BAAgB,SAAS,IAAI,CAAC;AAAA,QAChC;AACA,wBAAgB,SAAS,EAAE,KAAK,QAAQ,CAAC,CAAC;AAAA,MAC5C,OAAO;AACL,YAAI,CAAC,gBAAgB,IAAI,GAAG;AAC1B,0BAAgB,IAAI,IAAI,CAAC;AAAA,QAC3B;AAEA,wBAAgB,IAAI,EAAE,QAAQ,QAAQ,CAAC,CAAC;AAAA,MAC1C;AAAA,IACF;AAEA,UAAM,OAAO,OAAO,KAAK,eAAe;AACxC,aAAS,IAAI,GAAG,IAAI,KAAK,QAAQ,KAAK;AACpC,YAAM,OAAO,gBAAgB,KAAK,CAAC,CAAC;AAKpC,UAAI,OAAO,KAAK,CAAC,EAAE,SAAS,UAAU;AACpC,cAAM,OAAO,KAAK,CAAC;AACnB,cAAM,OAAO,KAAK;AAClB,YAAI,KAAK,KAAK,WAAW,MAAM,GAAG;AAChC,gBAAMA,QAAO,KAAK,SAAS,OAAO,EAAE,KAAK,gBAAgB,CAAC;AAC1D,gBAAM,OAAOA,MAAK,SAAS,KAAK;AAAA,YAC9B,KAAK;AAAA,YACL,MAAM,KAAK;AAAA,YACX,OAAO,KAAK;AAAA,UACd,CAAC;AACD,eAAK,YAAY,KAAK,yBAAyB,IAAI;AACnD,UAAAA,MAAK,QAAQ,aAAa,MAAM;AAChC;AAAA,QACF;AAAA,MACF;AAIA,UAAI;AACJ,YAAM,sBAAsB,KAAK,MAAM,KAAK,CAAC,EAAE,aAAa,GAAG,IAAI;AACnE,UAAI,KAAK,SAAS,gBAAgB;AAChC,cAAM,MAAM,KAAK,CAAC,EAAE,KAAK,MAAM,GAAG;AAClC,yBAAiB,IAAI,IAAI,SAAS,CAAC;AACnC,cAAM,OAAO,IAAI,MAAM,GAAG,IAAI,SAAS,CAAC,EAAE,KAAK,GAAG;AAClD,yBAAiB,UAAU,UAAU,kCAAkC;AAAA,MACzE,OAAO;AACL,yBAAiB,KAAK,CAAC,EAAE,KAAK,MAAM,GAAG,EAAE,IAAI;AAE7C,0BAAkB,QAAQ;AAAA,MAC5B;AAMA,UAAG,CAAC,KAAK,qBAAqB,KAAK,CAAC,EAAE,IAAI,GAAG;AAC3C,cAAMA,QAAO,KAAK,SAAS,OAAO,EAAE,KAAK,gBAAgB,CAAC;AAC1D,cAAME,aAAYF,MAAK,SAAS,KAAK;AAAA,UACnC,KAAK;AAAA,UACL,OAAO,KAAK,CAAC,EAAE;AAAA,QACjB,CAAC;AACD,QAAAE,WAAU,YAAY;AAEtB,aAAK,mBAAmBA,YAAW,KAAK,CAAC,GAAGF,KAAI;AAChD;AAAA,MACF;AAIA,uBAAiB,eAAe,QAAQ,OAAO,EAAE,EAAE,QAAQ,MAAM,KAAK;AACtE,YAAM,OAAO,KAAK,SAAS,OAAO,EAAE,KAAK,oBAAoB,CAAC;AAC9D,YAAM,SAAS,KAAK,SAAS,QAAQ,EAAE,KAAK,eAAe,CAAC;AAE5D,eAAS,QAAQ,QAAQ,gBAAgB;AACzC,YAAM,YAAY,OAAO,SAAS,KAAK;AAAA,QACrC,KAAK;AAAA,QACL,OAAO,KAAK,CAAC,EAAE;AAAA,MACjB,CAAC;AACD,gBAAU,YAAY;AAEtB,WAAK,mBAAmB,WAAW,KAAK,CAAC,GAAG,MAAM;AAClD,aAAO,iBAAiB,SAAS,CAAC,UAAU;AAE1C,YAAI,SAAS,MAAM;AACnB,eAAO,CAAC,OAAO,UAAU,SAAS,eAAe,GAAG;AAClD,mBAAS,OAAO;AAAA,QAClB;AACA,eAAO,UAAU,OAAO,cAAc;AAAA,MAExC,CAAC;AACD,YAAM,iBAAiB,KAAK,SAAS,IAAI;AAEzC,eAAS,IAAI,GAAG,IAAI,KAAK,QAAQ,KAAK;AAEpC,YAAG,KAAK,CAAC,EAAE,KAAK,QAAQ,GAAG,IAAI,IAAI;AACjC,gBAAM,QAAQ,KAAK,CAAC;AACpB,gBAAM,aAAa,eAAe,SAAS,MAAM;AAAA,YAC/C,KAAK;AAAA,YACL,OAAO,MAAM;AAAA,UACf,CAAC;AAED,cAAG,KAAK,SAAS,GAAG;AAClB,kBAAM,gBAAgB,KAAK,qBAAqB,KAAK;AACrD,kBAAM,uBAAuB,KAAK,MAAM,MAAM,aAAa,GAAG,IAAI;AAClE,uBAAW,YAAY,UAAU,mBAAmB;AAAA,UACtD;AACA,gBAAM,kBAAkB,WAAW,SAAS,KAAK;AAEjD,mBAAS,iBAAiB,eAAgB,MAAM,KAAK,gBAAgB,MAAM,MAAM,EAAC,OAAO,IAAI,WAAW,IAAI,CAAC,GAAI,iBAAiB,MAAM,MAAM,IAAI,SAAS,UAAU,CAAC;AAEtK,eAAK,mBAAmB,YAAY,OAAO,cAAc;AAAA,QAC3D,OAAK;AAEH,gBAAMG,kBAAiB,KAAK,SAAS,IAAI;AACzC,gBAAM,aAAaA,gBAAe,SAAS,MAAM;AAAA,YAC/C,KAAK;AAAA,YACL,OAAO,KAAK,CAAC,EAAE;AAAA,UACjB,CAAC;AACD,gBAAM,kBAAkB,WAAW,SAAS,KAAK;AACjD,cAAI,kBAAkB,MAAM,KAAK,eAAe,KAAK,CAAC,EAAE,MAAM,EAAC,OAAO,IAAI,WAAW,IAAI,CAAC;AAC1F,cAAG,CAAC;AAAiB;AACrB,mBAAS,iBAAiB,eAAe,iBAAiB,iBAAiB,KAAK,CAAC,EAAE,MAAM,IAAI,SAAS,UAAU,CAAC;AACjH,eAAK,mBAAmB,YAAY,KAAK,CAAC,GAAGA,eAAc;AAAA,QAE7D;AAAA,MACF;AAAA,IACF;AACA,SAAK,aAAa,WAAW,MAAM;AAAA,EACrC;AAAA,EAEA,mBAAmB,MAAM,MAAM,MAAM;AACnC,SAAK,iBAAiB,SAAS,OAAO,UAAU;AAC9C,YAAM,KAAK,UAAU,MAAM,KAAK;AAAA,IAClC,CAAC;AAGD,SAAK,QAAQ,aAAa,MAAM;AAChC,SAAK,iBAAiB,aAAa,CAAC,UAAU;AAC5C,YAAM,cAAc,KAAK,IAAI;AAC7B,YAAM,YAAY,KAAK,KAAK,MAAM,GAAG,EAAE,CAAC;AACxC,YAAM,OAAO,KAAK,IAAI,cAAc,qBAAqB,WAAW,EAAE;AACtE,YAAM,WAAW,YAAY,SAAS,OAAO,IAAI;AAEjD,kBAAY,YAAY,OAAO,QAAQ;AAAA,IACzC,CAAC;AAED,QAAI,KAAK,KAAK,QAAQ,GAAG,IAAI;AAAI;AAEjC,SAAK,iBAAiB,aAAa,CAAC,UAAU;AAC5C,WAAK,IAAI,UAAU,QAAQ,cAAc;AAAA,QACvC;AAAA,QACA,QAAQ;AAAA,QACR,aAAa;AAAA,QACb,UAAU;AAAA,QACV,UAAU,KAAK;AAAA,MACjB,CAAC;AAAA,IACH,CAAC;AAAA,EACH;AAAA;AAAA;AAAA,EAIA,MAAM,UAAU,MAAM,QAAM,MAAM;AAChC,QAAI;AACJ,QAAI;AACJ,QAAI,KAAK,KAAK,QAAQ,GAAG,IAAI,IAAI;AAE/B,mBAAa,KAAK,IAAI,cAAc,qBAAqB,KAAK,KAAK,MAAM,GAAG,EAAE,CAAC,GAAG,EAAE;AAEpF,YAAM,oBAAoB,KAAK,IAAI,cAAc,aAAa,UAAU;AAGxE,UAAI,eAAe,KAAK,KAAK,MAAM,GAAG,EAAE,IAAI;AAE5C,UAAI,YAAY;AAChB,UAAI,aAAa,QAAQ,GAAG,IAAI,IAAI;AAElC,oBAAY,SAAS,aAAa,MAAM,GAAG,EAAE,CAAC,EAAE,MAAM,GAAG,EAAE,CAAC,CAAC;AAE7D,uBAAe,aAAa,MAAM,GAAG,EAAE,CAAC;AAAA,MAC1C;AAEA,YAAM,WAAW,kBAAkB;AAEnC,eAAQ,IAAI,GAAG,IAAI,SAAS,QAAQ,KAAK;AACvC,YAAI,SAAS,CAAC,EAAE,YAAY,cAAc;AAExC,cAAG,cAAc,GAAG;AAClB,sBAAU,SAAS,CAAC;AACpB;AAAA,UACF;AACA;AAAA,QACF;AAAA,MACF;AAAA,IAEF,OAAO;AACL,mBAAa,KAAK,IAAI,cAAc,qBAAqB,KAAK,MAAM,EAAE;AAAA,IACxE;AACA,QAAI;AACJ,QAAG,OAAO;AAER,YAAM,MAAM,SAAS,OAAO,WAAW,KAAK;AAE5C,aAAO,KAAK,IAAI,UAAU,QAAQ,GAAG;AAAA,IACvC,OAAK;AAEH,aAAO,KAAK,IAAI,UAAU,kBAAkB;AAAA,IAC9C;AACA,UAAM,KAAK,SAAS,UAAU;AAC9B,QAAI,SAAS;AACX,UAAI,EAAE,OAAO,IAAI,KAAK;AACtB,YAAM,MAAM,EAAE,MAAM,QAAQ,SAAS,MAAM,MAAM,IAAI,EAAE;AACvD,aAAO,UAAU,GAAG;AACpB,aAAO,eAAe,EAAE,IAAI,KAAK,MAAM,IAAI,GAAG,IAAI;AAAA,IACpD;AAAA,EACF;AAAA,EAEA,qBAAqB,OAAO;AAC1B,UAAM,iBAAiB,MAAM,KAAK,MAAM,KAAK,EAAE,CAAC,EAAE,MAAM,GAAG;AAE3D,QAAI,gBAAgB;AACpB,aAAS,IAAI,eAAe,SAAS,GAAG,KAAK,GAAG,KAAK;AACnD,UAAG,cAAc,SAAS,GAAG;AAC3B,wBAAgB,MAAM;AAAA,MACxB;AACA,sBAAgB,eAAe,CAAC,IAAI;AAEpC,UAAI,cAAc,SAAS,KAAK;AAC9B;AAAA,MACF;AAAA,IACF;AAEA,QAAI,cAAc,WAAW,KAAK,GAAG;AACnC,sBAAgB,cAAc,MAAM,CAAC;AAAA,IACvC;AACA,WAAO;AAAA,EAET;AAAA,EAEA,qBAAqB,MAAM;AACzB,WAAQ,KAAK,QAAQ,KAAK,MAAM,MAAQ,KAAK,QAAQ,aAAa,MAAM;AAAA,EAC1E;AAAA,EAEA,yBAAyB,MAAK;AAC5B,QAAG,KAAK,QAAQ;AACd,UAAG,KAAK,WAAW;AAAS,aAAK,SAAS;AAC1C,aAAO,UAAU,KAAK,qBAAqB,KAAK;AAAA,IAClD;AAEA,QAAI,SAAS,KAAK,KAAK,QAAQ,iBAAiB,EAAE;AAElD,aAAS,OAAO,MAAM,GAAG,EAAE,CAAC;AAE5B,WAAO,oBAAa,qBAAqB,KAAK;AAAA,EAChD;AAAA;AAAA,EAEA,MAAM,kBAAkB;AACtB,QAAG,CAAC,KAAK,WAAW,KAAK,QAAQ,WAAW,GAAE;AAC5C,WAAK,UAAU,MAAM,KAAK,YAAY;AAAA,IACxC;AACA,WAAO,KAAK;AAAA,EACd;AAAA;AAAA,EAEA,MAAM,YAAY,OAAO,KAAK;AAC5B,QAAI,WAAW,MAAM,KAAK,IAAI,MAAM,QAAQ,KAAK,IAAI,GAAG;AACxD,QAAI,cAAc,CAAC;AACnB,aAAS,IAAI,GAAG,IAAI,QAAQ,QAAQ,KAAK;AACvC,UAAI,QAAQ,CAAC,EAAE,WAAW,GAAG;AAAG;AAChC,kBAAY,KAAK,QAAQ,CAAC,CAAC;AAC3B,oBAAc,YAAY,OAAO,MAAM,KAAK,YAAY,QAAQ,CAAC,IAAI,GAAG,CAAC;AAAA,IAC3E;AACA,WAAO;AAAA,EACT;AAAA,EAGA,MAAM,aAAa;AAEjB,QAAG,CAAC,KAAK,SAAS,aAAY;AAC5B,UAAI,SAAS,OAAO,kGAAkG;AACtH;AAAA,IACF;AACA,YAAQ,IAAI,eAAe;AAE3B,UAAM,QAAQ,KAAK,IAAI,MAAM,iBAAiB,EAAE,OAAO,CAAC,SAAS;AAE/D,eAAQ,IAAI,GAAG,IAAI,KAAK,gBAAgB,QAAQ,KAAK;AACnD,YAAG,KAAK,KAAK,QAAQ,KAAK,gBAAgB,CAAC,CAAC,IAAI,IAAI;AAClD,iBAAO;AAAA,QACT;AAAA,MACF;AACA,aAAO;AAAA,IACT,CAAC;AACD,UAAM,QAAQ,MAAM,KAAK,mBAAmB,KAAK;AACjD,YAAQ,IAAI,cAAc;AAE1B,UAAM,KAAK,IAAI,MAAM,QAAQ,MAAM,iCAAiC,KAAK,UAAU,OAAO,MAAM,CAAC,CAAC;AAClG,YAAQ,IAAI,aAAa;AACzB,YAAQ,IAAI,KAAK,SAAS,WAAW;AAErC,UAAM,WAAW,OAAO,GAAG,SAAS,YAAY;AAAA,MAC9C,KAAK;AAAA,MACL,QAAQ;AAAA,MACR,SAAS;AAAA,QACP,gBAAgB;AAAA,MAClB;AAAA,MACA,aAAa;AAAA,MACb,MAAM,KAAK,UAAU;AAAA,QACnB,aAAa,KAAK,SAAS;AAAA,QAC3B;AAAA,MACF,CAAC;AAAA,IACH,CAAC;AACD,YAAQ,IAAI,QAAQ;AAAA,EAEtB;AAAA,EAEA,MAAM,mBAAmB,OAAO;AAC9B,QAAI,SAAS,CAAC;AAEd,aAAQ,IAAI,GAAG,IAAI,MAAM,QAAQ,KAAK;AACpC,UAAI,OAAO,MAAM,CAAC;AAClB,UAAI,QAAQ,KAAK,KAAK,MAAM,GAAG;AAC/B,UAAI,UAAU;AAEd,eAAS,KAAK,GAAG,KAAK,MAAM,QAAQ,MAAM;AACxC,YAAI,OAAO,MAAM,EAAE;AAEnB,YAAI,OAAO,MAAM,SAAS,GAAG;AAE3B,kBAAQ,IAAI,IAAI,MAAM,KAAK,IAAI,MAAM,WAAW,IAAI;AAAA,QACtD,OAAO;AAEL,cAAI,CAAC,QAAQ,IAAI,GAAG;AAClB,oBAAQ,IAAI,IAAI,CAAC;AAAA,UACnB;AAEA,oBAAU,QAAQ,IAAI;AAAA,QACxB;AAAA,MACF;AAAA,IACF;AAEA,WAAO;AAAA,EACT;AAEF;AAEA,IAAM,8BAA8B;AACpC,IAAM,uBAAN,cAAmC,SAAS,SAAS;AAAA,EACnD,YAAY,MAAM,QAAQ;AACxB,UAAM,IAAI;AACV,SAAK,SAAS;AACd,SAAK,UAAU;AACf,SAAK,YAAY;AAAA,EACnB;AAAA,EACA,cAAc;AACZ,WAAO;AAAA,EACT;AAAA,EAEA,iBAAiB;AACf,WAAO;AAAA,EACT;AAAA,EAEA,UAAU;AACR,WAAO;AAAA,EACT;AAAA,EAGA,YAAY,SAAS;AACnB,UAAM,YAAY,KAAK,YAAY,SAAS,CAAC;AAE7C,cAAU,MAAM;AAEhB,SAAK,iBAAiB,SAAS;AAE/B,QAAI,MAAM,QAAQ,OAAO,GAAG;AAC1B,eAAS,IAAI,GAAG,IAAI,QAAQ,QAAQ,KAAK;AACvC,kBAAU,SAAS,KAAK,EAAE,KAAK,cAAc,MAAM,QAAQ,CAAC,EAAE,CAAC;AAAA,MACjE;AAAA,IACF,OAAK;AAEH,gBAAU,SAAS,KAAK,EAAE,KAAK,cAAc,MAAM,QAAQ,CAAC;AAAA,IAC9D;AAAA,EACF;AAAA,EACA,iBAAiB,MAAM,iBAAe,OAAO;AAK3C,QAAI,CAAC,gBAAgB;AACnB,aAAO,KAAK,MAAM,GAAG,EAAE,IAAI;AAAA,IAC7B;AAEA,QAAI,KAAK,QAAQ,GAAG,IAAI,IAAI;AAE1B,aAAO,KAAK,MAAM,KAAK;AAEvB,WAAK,CAAC,IAAI,UAAU,KAAK,CAAC;AAE1B,aAAO,KAAK,KAAK,EAAE;AAEnB,aAAO,KAAK,QAAQ,OAAO,QAAK;AAAA,IAClC,OAAK;AAEH,aAAO,KAAK,QAAQ,OAAO,EAAE;AAAA,IAC/B;AACA,WAAO;AAAA,EACT;AAAA,EAGA,YAAY,SAAS,kBAAgB,MAAM,eAAa,OAAO;AAE7D,UAAM,YAAY,KAAK,YAAY,SAAS,CAAC;AAE7C,QAAG,CAAC,cAAa;AAEf,gBAAU,MAAM;AAChB,WAAK,iBAAiB,WAAW,eAAe;AAAA,IAClD;AAEA,SAAK,OAAO,eAAe,WAAW,OAAO;AAAA,EAC/C;AAAA,EAEA,iBAAiB,WAAW,kBAAgB,MAAM;AAChD,QAAI;AAEJ,QAAK,UAAU,SAAS,SAAS,KAAO,UAAU,SAAS,CAAC,EAAE,UAAU,SAAS,YAAY,GAAI;AAC/F,gBAAU,UAAU,SAAS,CAAC;AAC9B,cAAQ,MAAM;AAAA,IAChB,OAAO;AAEL,gBAAU,UAAU,SAAS,OAAO,EAAE,KAAK,aAAa,CAAC;AAAA,IAC3D;AAEA,QAAI,iBAAiB;AACnB,cAAQ,SAAS,KAAK,EAAE,KAAK,cAAc,MAAM,gBAAgB,CAAC;AAAA,IACpE;AAEA,UAAM,cAAc,QAAQ,SAAS,UAAU,EAAE,KAAK,iBAAiB,CAAC;AAExE,aAAS,QAAQ,aAAa,gBAAgB;AAE9C,gBAAY,iBAAiB,SAAS,MAAM;AAE1C,WAAK,OAAO,UAAU;AAAA,IACxB,CAAC;AAED,UAAM,gBAAgB,QAAQ,SAAS,UAAU,EAAE,KAAK,mBAAmB,CAAC;AAE5E,aAAS,QAAQ,eAAe,QAAQ;AAExC,kBAAc,iBAAiB,SAAS,MAAM;AAE5C,cAAQ,MAAM;AAEd,YAAM,mBAAmB,QAAQ,SAAS,OAAO,EAAE,KAAK,yBAAyB,CAAC;AAClF,YAAM,QAAQ,iBAAiB,SAAS,SAAS;AAAA,QAC/C,KAAK;AAAA,QACL,MAAM;AAAA,QACN,aAAa;AAAA,MACf,CAAC;AAED,YAAM,MAAM;AAEZ,YAAM,iBAAiB,WAAW,CAAC,UAAU;AAE3C,YAAI,MAAM,QAAQ,UAAU;AAC1B,eAAK,oBAAoB;AAEzB,eAAK,iBAAiB,WAAW,eAAe;AAAA,QAClD;AAAA,MACF,CAAC;AAGD,YAAM,iBAAiB,SAAS,CAAC,UAAU;AAEzC,aAAK,oBAAoB;AAEzB,cAAM,cAAc,MAAM;AAE1B,YAAI,MAAM,QAAQ,WAAW,gBAAgB,IAAI;AAC/C,eAAK,OAAO,WAAW;AAAA,QACzB,WAES,gBAAgB,IAAI;AAE3B,uBAAa,KAAK,cAAc;AAEhC,eAAK,iBAAiB,WAAW,MAAM;AACrC,iBAAK,OAAO,aAAa,IAAI;AAAA,UAC/B,GAAG,GAAG;AAAA,QACR;AAAA,MACF,CAAC;AAAA,IACH,CAAC;AAAA,EACH;AAAA;AAAA,EAGA,4BAA4B;AAE1B,UAAM,YAAY,KAAK,YAAY,SAAS,CAAC;AAE7C,cAAU,MAAM;AAEhB,cAAU,SAAS,MAAM,EAAE,KAAK,aAAa,MAAM,4BAA4B,CAAC;AAEhF,UAAM,aAAa,UAAU,SAAS,OAAO,EAAE,KAAK,cAAc,CAAC;AAEnE,UAAM,gBAAgB,WAAW,SAAS,UAAU,EAAE,KAAK,YAAY,MAAM,yBAAyB,CAAC;AAEvG,eAAW,SAAS,KAAK,EAAE,KAAK,gBAAgB,MAAM,0FAA0F,CAAC;AAEjJ,UAAM,eAAe,WAAW,SAAS,UAAU,EAAE,KAAK,YAAY,MAAM,QAAQ,CAAC;AAErF,eAAW,SAAS,KAAK,EAAE,KAAK,gBAAgB,MAAM,mEAAmE,CAAC;AAG1H,kBAAc,iBAAiB,SAAS,OAAO,UAAU;AAEvD,YAAM,KAAK,OAAO,eAAe,qBAAqB;AAEtD,YAAM,KAAK,mBAAmB;AAAA,IAChC,CAAC;AAGD,iBAAa,iBAAiB,SAAS,OAAO,UAAU;AACtD,cAAQ,IAAI,uCAAuC;AAEnD,YAAM,KAAK,OAAO,UAAU;AAE5B,YAAM,KAAK,mBAAmB;AAAA,IAChC,CAAC;AAAA,EACH;AAAA,EAEA,MAAM,SAAS;AACb,UAAM,YAAY,KAAK,YAAY,SAAS,CAAC;AAC7C,cAAU,MAAM;AAEhB,cAAU,SAAS,KAAK,EAAE,KAAK,iBAAiB,MAAM,mCAAmC,CAAC;AAG1F,SAAK,OAAO,cAAc,KAAK,IAAI,UAAU,GAAG,aAAa,CAAC,SAAS;AAErE,UAAG,CAAC,MAAM;AAER;AAAA,MACF;AAEA,UAAG,qBAAqB,QAAQ,KAAK,SAAS,MAAM,IAAI;AACtD,eAAO,KAAK,YAAY;AAAA,UACtB,WAAS,KAAK;AAAA,UACb,uCAAqC,qBAAqB,KAAK,IAAI,IAAE;AAAA,QACxE,CAAC;AAAA,MACH;AAEA,UAAG,KAAK,WAAU;AAChB,qBAAa,KAAK,SAAS;AAAA,MAC7B;AACA,WAAK,YAAY,WAAW,MAAM;AAChC,aAAK,mBAAmB,IAAI;AAC5B,aAAK,YAAY;AAAA,MACnB,GAAG,GAAI;AAAA,IAET,CAAC,CAAC;AAEF,SAAK,IAAI,UAAU,wBAAwB,6BAA6B;AAAA,MACpE,SAAS;AAAA,MACT,YAAY;AAAA,IAChB,CAAC;AACD,SAAK,IAAI,UAAU,wBAAwB,kCAAkC;AAAA,MACzE,SAAS;AAAA,MACT,YAAY;AAAA,IAChB,CAAC;AAED,SAAK,IAAI,UAAU,cAAc,KAAK,WAAW,KAAK,IAAI,CAAC;AAAA,EAE7D;AAAA,EAEA,MAAM,aAAa;AACjB,SAAK,YAAY,4BAA4B;AAC7C,UAAM,gBAAgB,MAAM,KAAK,OAAO,UAAU;AAClD,QAAG,eAAc;AACf,WAAK,YAAY,yBAAyB;AAC1C,YAAM,KAAK,mBAAmB;AAAA,IAChC,OAAK;AACH,WAAK,0BAA0B;AAAA,IACjC;AAOA,SAAK,MAAM,IAAI,wBAAwB,KAAK,KAAK,KAAK,QAAQ,IAAI;AAElE,KAAC,OAAO,yBAAyB,IAAI,KAAK,QAAQ,KAAK,SAAS,MAAM,OAAO,OAAO,yBAAyB,CAAC;AAAA,EAEhH;AAAA,EAEA,MAAM,UAAU;AACd,YAAQ,IAAI,gCAAgC;AAC5C,SAAK,IAAI,UAAU,0BAA0B,2BAA2B;AACxE,SAAK,OAAO,OAAO;AAAA,EACrB;AAAA,EAEA,MAAM,mBAAmB,UAAQ,MAAM;AACrC,YAAQ,IAAI,uBAAuB;AAEnC,QAAG,CAAC,KAAK,OAAO,SAAS,SAAS;AAChC,WAAK,YAAY,yDAAyD;AAC1E;AAAA,IACF;AACA,QAAG,CAAC,KAAK,OAAO,mBAAkB;AAChC,YAAM,KAAK,OAAO,UAAU;AAAA,IAC9B;AAEA,QAAG,CAAC,KAAK,OAAO,mBAAmB;AACjC,cAAQ,IAAI,wDAAwD;AACpE,WAAK,0BAA0B;AAC/B;AAAA,IACF;AACA,SAAK,YAAY,6BAA6B;AAI9C,QAAG,OAAO,YAAY,UAAU;AAC9B,YAAM,mBAAmB;AAEzB,YAAM,KAAK,OAAO,gBAAgB;AAClC;AAAA,IACF;AAKA,SAAK,UAAU;AACf,SAAK,iBAAiB;AACtB,SAAK,YAAY;AACjB,SAAK,OAAO;AAEZ,QAAG,KAAK,UAAU;AAChB,oBAAc,KAAK,QAAQ;AAC3B,WAAK,WAAW;AAAA,IAClB;AAEA,SAAK,WAAW,YAAY,MAAM;AAChC,UAAG,CAAC,KAAK,WAAU;AACjB,YAAG,KAAK,gBAAgB,SAAS,OAAO;AACtC,eAAK,YAAY;AACjB,eAAK,wBAAwB,KAAK,IAAI;AAAA,QACxC,OAAK;AAEH,eAAK,OAAO,KAAK,IAAI,UAAU,cAAc;AAE7C,cAAG,CAAC,KAAK,QAAQ,KAAK,QAAQ,GAAG;AAC/B,0BAAc,KAAK,QAAQ;AAC3B,iBAAK,YAAY,gBAAgB;AACjC;AAAA,UACF;AAAA,QACF;AAAA,MACF,OAAK;AACH,YAAG,KAAK,SAAS;AACf,wBAAc,KAAK,QAAQ;AAE3B,cAAI,OAAO,KAAK,YAAY,UAAU;AACpC,iBAAK,YAAY,KAAK,OAAO;AAAA,UAC/B,OAAO;AAEL,iBAAK,YAAY,KAAK,SAAS,WAAW,KAAK,KAAK,IAAI;AAAA,UAC1D;AAEA,cAAI,KAAK,OAAO,WAAW,kBAAkB,SAAS,GAAG;AACvD,iBAAK,OAAO,uBAAuB;AAAA,UACrC;AAEA,eAAK,OAAO,kBAAkB;AAC9B;AAAA,QACF,OAAK;AACH,eAAK;AACL,eAAK,YAAY,gCAA8B,KAAK,cAAc;AAAA,QACpE;AAAA,MACF;AAAA,IACF,GAAG,EAAE;AAAA,EACP;AAAA,EAEA,MAAM,wBAAwB,MAAM;AAClC,SAAK,UAAU,MAAM,KAAK,OAAO,sBAAsB,IAAI;AAAA,EAC7D;AAAA,EAEA,sBAAsB;AACpB,QAAI,KAAK,gBAAgB;AACvB,mBAAa,KAAK,cAAc;AAChC,WAAK,iBAAiB;AAAA,IACxB;AAAA,EACF;AAAA,EAEA,MAAM,OAAO,aAAa,eAAa,OAAO;AAC5C,UAAM,UAAU,MAAM,KAAK,OAAO,IAAI,OAAO,WAAW;AAExD,UAAM,kBAAkB,eAAe,YAAY,SAAS,MAAM,YAAY,UAAU,GAAG,GAAG,IAAI,QAAQ;AAC1G,SAAK,YAAY,SAAS,iBAAiB,YAAY;AAAA,EACzD;AAEF;AACA,IAAM,0BAAN,MAA8B;AAAA,EAC5B,YAAY,KAAK,QAAQ,MAAM;AAC7B,SAAK,MAAM;AACX,SAAK,SAAS;AACd,SAAK,OAAO;AAAA,EACd;AAAA,EACA,MAAM,OAAO,aAAa;AACxB,WAAO,MAAM,KAAK,OAAO,IAAI,OAAO,WAAW;AAAA,EACjD;AAAA;AAAA,EAEA,MAAM,yBAAyB;AAC7B,UAAM,KAAK,OAAO,UAAU;AAC5B,UAAM,KAAK,KAAK,mBAAmB;AAAA,EACrC;AAAA,EACA,MAAM,YAAY;AAChB,SAAK,iBAAiB,IAAI,QAAQ;AAAA,MAChC,aAAa;AAAA,MACb,gBAAgB,KAAK,IAAI,MAAM,QAAQ,OAAO;AAAA,QAC5C,KAAK,IAAI,MAAM;AAAA,MACjB;AAAA,MACA,eAAe,KAAK,IAAI,MAAM,QAAQ,MAAM,KAAK,KAAK,IAAI,MAAM,OAAO;AAAA,MACvE,cAAc,KAAK,IAAI,MAAM,QAAQ,KAAK,KAAK,KAAK,IAAI,MAAM,OAAO;AAAA,MACrE,gBAAgB,KAAK,IAAI,MAAM,QAAQ,OAAO;AAAA,QAC5C,KAAK,IAAI,MAAM;AAAA,MACjB;AAAA,MACA,cAAc,KAAK,IAAI,MAAM,QAAQ,KAAK,KAAK,KAAK,IAAI,MAAM,OAAO;AAAA,MACrE,eAAe,KAAK,IAAI,MAAM,QAAQ,MAAM,KAAK,KAAK,IAAI,MAAM,OAAO;AAAA,IACzE,CAAC;AACD,SAAK,oBAAoB,MAAM,KAAK,eAAe,KAAK;AACxD,WAAO,KAAK;AAAA,EACd;AACF;AACA,IAAM,cAAN,MAAkB;AAAA,EAChB,YAAY,KAAK,QAAQ;AACvB,SAAK,MAAM;AACX,SAAK,SAAS;AAAA,EAChB;AAAA,EACA,MAAM,OAAQ,aAAa,SAAO,CAAC,GAAG;AACpC,aAAS;AAAA,MACP,eAAe,KAAK,OAAO,SAAS;AAAA,MACpC,GAAG;AAAA,IACL;AACA,QAAI,UAAU,CAAC;AACf,UAAM,OAAO,MAAM,KAAK,OAAO,6BAA6B,WAAW;AACvE,QAAI,QAAQ,KAAK,QAAQ,KAAK,KAAK,CAAC,KAAK,KAAK,KAAK,CAAC,EAAE,WAAW;AAC/D,gBAAU,KAAK,OAAO,eAAe,QAAQ,KAAK,KAAK,CAAC,EAAE,WAAW,MAAM;AAAA,IAC7E,OAAO;AAEL,UAAI,SAAS,OAAO,4CAA4C;AAAA,IAClE;AACA,WAAO;AAAA,EACT;AACF;AAEA,IAAM,8BAAN,cAA0C,SAAS,iBAAiB;AAAA,EAClE,YAAY,KAAK,QAAQ;AACvB,UAAM,KAAK,MAAM;AACjB,SAAK,SAAS;AAAA,EAChB;AAAA,EACA,UAAU;AACR,UAAM;AAAA,MACJ;AAAA,IACF,IAAI;AACJ,gBAAY,MAAM;AAClB,gBAAY,SAAS,MAAM;AAAA,MACzB,MAAM;AAAA,IACR,CAAC;AAED,gBAAY,SAAS,KAAK;AAAA,MACxB,MAAM;AAAA,IACR,CAAC;AAED,UAAM,0BAA0B,YAAY,SAAS,IAAI;AACzD,4BAAwB,SAAS,MAAM;AAAA,MACrC,MAAM;AAAA,IACR,CAAC;AACD,4BAAwB,SAAS,MAAM;AAAA,MACrC,MAAM;AAAA,IACR,CAAC;AACD,4BAAwB,SAAS,MAAM;AAAA,MACrC,MAAM;AAAA,IACR,CAAC;AAED,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,uBAAuB,EAAE,QAAQ,sDAAsD,EAAE,QAAQ,CAAC,SAAS,KAAK,eAAe,wBAAwB,EAAE,SAAS,KAAK,OAAO,SAAS,WAAW,EAAE,SAAS,OAAO,UAAU;AACtQ,WAAK,OAAO,SAAS,cAAc,MAAM,KAAK;AAC9C,YAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IACrC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,QAAQ,EAAE,QAAQ,mIAAmI,EAAE,UAAU,CAAC,WAAW,OAAO,cAAc,mBAAmB,EAAE,QAAQ,YAAY;AACnR,YAAM,KAAK,OAAO,aAAa;AAAA,IACjC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,YAAY,EAAE,QAAQ,8GAA8G,EAAE,UAAU,CAAC,WAAW,OAAO,cAAc,YAAY,EAAE,QAAQ,YAAY;AAE3P,YAAM,KAAK,OAAO,WAAW;AAAA,IAC/B,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,oBAAoB,EAAE,QAAQ,oBAAoB,EAAE,UAAU,CAAC,WAAW,OAAO,cAAc,oBAAoB,EAAE,QAAQ,YAAY;AACjL,YAAM,gBAAgB;AAAA,QACpB;AAAA,QACA;AAAA,MACF;AACA,UAAG,CAAC,KAAK,OAAO,oBAAmB;AACjC,aAAK,OAAO,qBAAqB,KAAK,MAAM,KAAK,OAAO,CAAC;AAAA,MAC3D;AAEA,aAAO,KAAK,cAAc,KAAK,OAAO,kBAAkB,CAAC;AAAA,IAC3D,CAAC,CAAC;AAGF,gBAAY,SAAS,MAAM;AAAA,MACzB,MAAM;AAAA,IACR,CAAC;AAED,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,gBAAgB,EAAE,QAAQ,6EAA6E,EAAE,QAAQ,CAAC,SAAS,KAAK,eAAe,oBAAoB,EAAE,SAAS,KAAK,OAAO,SAAS,OAAO,EAAE,SAAS,OAAO,UAAU;AAC9Q,WAAK,OAAO,SAAS,UAAU,MAAM,KAAK;AAC1C,YAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IACrC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,cAAc,EAAE,QAAQ,cAAc,EAAE,UAAU,CAAC,WAAW,OAAO,cAAc,cAAc,EAAE,QAAQ,YAAY;AAE/J,YAAM,OAAO,MAAM,KAAK,OAAO,aAAa;AAC5C,UAAG,MAAM;AACP,YAAI,SAAS,OAAO,qCAAqC;AAAA,MAC3D,OAAK;AACH,YAAI,SAAS,OAAO,wDAAwD;AAAA,MAC9E;AAAA,IACF,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,kBAAkB,EAAE,QAAQ,wCAAwC,EAAE,YAAY,CAAC,aAAa;AACxI,eAAS,UAAU,qBAAqB,mBAAmB;AAC3D,eAAS,UAAU,SAAS,4BAA4B;AACxD,eAAS,UAAU,iBAAiB,oBAAoB;AACxD,eAAS,UAAU,sBAAsB,oBAAoB;AAC7D,eAAS,SAAS,OAAO,UAAU;AACjC,aAAK,OAAO,SAAS,mBAAmB;AACxC,cAAM,KAAK,OAAO,aAAa;AAAA,MACjC,CAAC;AACD,eAAS,SAAS,KAAK,OAAO,SAAS,gBAAgB;AAAA,IACzD,CAAC;AAED,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,kBAAkB,EAAE,QAAQ,oHAAoH,EAAE,YAAY,CAAC,aAAa;AAEpN,YAAM,YAAY,OAAO,KAAK,iBAAiB;AAC/C,eAAQ,IAAI,GAAG,IAAI,UAAU,QAAQ,KAAK;AACxC,iBAAS,UAAU,UAAU,CAAC,GAAG,UAAU,CAAC,CAAC;AAAA,MAC/C;AACA,eAAS,SAAS,OAAO,UAAU;AACjC,aAAK,OAAO,SAAS,WAAW;AAChC,cAAM,KAAK,OAAO,aAAa;AAC/B,+BAAuB,QAAQ,KAAK,kBAAkB,CAAC;AAEvD,cAAM,YAAY,KAAK,IAAI,UAAU,gBAAgB,gCAAgC,EAAE,SAAS,IAAI,KAAK,IAAI,UAAU,gBAAgB,gCAAgC,EAAE,CAAC,EAAE,OAAO;AACnL,YAAG,WAAW;AACZ,oBAAU,SAAS;AAAA,QACrB;AAAA,MACF,CAAC;AACD,eAAS,SAAS,KAAK,OAAO,SAAS,QAAQ;AAAA,IACjD,CAAC;AAED,UAAM,yBAAyB,YAAY,SAAS,QAAQ;AAAA,MAC1D,MAAM,KAAK,kBAAkB;AAAA,IAC/B,CAAC;AACD,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,qBAAqB,EAAE,QAAQ,0EAA0E,EAAE,UAAU,CAAC,WAAW;AAAE,aAAO,SAAS,KAAK,OAAO,SAAS,mBAAmB,EAAE,SAAS,OAAO,UAAU;AAAE,aAAK,OAAO,SAAS,sBAAsB;AAAO,cAAM,KAAK,OAAO,aAAa;AAAA,MAAG,CAAC;AAAA,IAAG,CAAC;AAC5V,gBAAY,SAAS,MAAM;AAAA,MACzB,MAAM;AAAA,IACR,CAAC;AAED,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,iBAAiB,EAAE,QAAQ,gDAAgD,EAAE,QAAQ,CAAC,SAAS,KAAK,eAAe,uBAAuB,EAAE,SAAS,KAAK,OAAO,SAAS,eAAe,EAAE,SAAS,OAAO,UAAU;AAC7P,WAAK,OAAO,SAAS,kBAAkB;AACvC,YAAM,KAAK,OAAO,aAAa;AAAA,IACjC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,mBAAmB,EAAE,QAAQ,kDAAkD,EAAE,QAAQ,CAAC,SAAS,KAAK,eAAe,uBAAuB,EAAE,SAAS,KAAK,OAAO,SAAS,iBAAiB,EAAE,SAAS,OAAO,UAAU;AACnQ,WAAK,OAAO,SAAS,oBAAoB;AACzC,YAAM,KAAK,OAAO,aAAa;AAAA,IACjC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,WAAW,EAAE,QAAQ,4CAA4C,EAAE,QAAQ,CAAC,SAAS,KAAK,eAAe,uBAAuB,EAAE,SAAS,KAAK,OAAO,SAAS,SAAS,EAAE,SAAS,OAAO,UAAU;AAC7O,WAAK,OAAO,SAAS,YAAY;AACjC,YAAM,KAAK,OAAO,aAAa;AAAA,IACjC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,mBAAmB,EAAE,QAAQ,2EAA2E,EAAE,QAAQ,CAAC,SAAS,KAAK,eAAe,uBAAuB,EAAE,SAAS,KAAK,OAAO,SAAS,iBAAiB,EAAE,SAAS,OAAO,UAAU;AAC5R,WAAK,OAAO,SAAS,oBAAoB;AACzC,YAAM,KAAK,OAAO,aAAa;AAAA,IACjC,CAAC,CAAC;AACF,gBAAY,SAAS,MAAM;AAAA,MACzB,MAAM;AAAA,IACR,CAAC;AAED,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,gBAAgB,EAAE,QAAQ,yBAAyB,EAAE,UAAU,CAAC,WAAW,OAAO,SAAS,KAAK,OAAO,SAAS,cAAc,EAAE,SAAS,OAAO,UAAU;AAClM,WAAK,OAAO,SAAS,iBAAiB;AACtC,YAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IACrC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,eAAe,EAAE,QAAQ,2BAA2B,EAAE,UAAU,CAAC,WAAW,OAAO,SAAS,KAAK,OAAO,SAAS,aAAa,EAAE,SAAS,OAAO,UAAU;AAClM,WAAK,OAAO,SAAS,gBAAgB;AACrC,YAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IACrC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,uBAAuB,EAAE,QAAQ,wBAAwB,EAAE,UAAU,CAAC,WAAW,OAAO,SAAS,KAAK,OAAO,SAAS,qBAAqB,EAAE,SAAS,OAAO,UAAU;AAC/M,WAAK,OAAO,SAAS,wBAAwB;AAC7C,YAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IACrC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,WAAW,EAAE,QAAQ,gCAAgC,EAAE,UAAU,CAAC,WAAW,OAAO,SAAS,KAAK,OAAO,SAAS,SAAS,EAAE,SAAS,OAAO,UAAU;AAC/L,WAAK,OAAO,SAAS,YAAY;AACjC,YAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IACrC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,WAAW,EAAE,QAAQ,gCAAgC,EAAE,UAAU,CAAC,WAAW,OAAO,SAAS,KAAK,OAAO,SAAS,SAAS,EAAE,SAAS,OAAO,UAAU;AAC/L,WAAK,OAAO,SAAS,YAAY;AACjC,YAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IACrC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,kBAAkB,EAAE,QAAQ,iCAAiC,EAAE,YAAY,CAAC,aAAa;AACjI,eAAS,UAAU,OAAO,oBAAoB;AAC9C,eAAS,UAAU,MAAM,iBAAiB;AAC1C,eAAS,SAAS,KAAK,OAAO,SAAS,gBAAgB;AACvD,eAAS,SAAS,OAAO,UAAU;AACjC,aAAK,OAAO,SAAS,mBAAmB,KAAK,MAAM,KAAK;AACxD,cAAM,KAAK,OAAO,aAAa,IAAI;AACnC,aAAK,OAAO,UAAU;AAAA,MAExB,CAAC;AAAA,IACH,CAAC;AACD,gBAAY,SAAS,MAAM;AAAA,MACzB,MAAM;AAAA,IACR,CAAC;AAED,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,YAAY,EAAE,QAAQ,uDAAuD,EAAE,UAAU,CAAC,WAAW,OAAO,SAAS,KAAK,OAAO,SAAS,UAAU,EAAE,SAAS,OAAO,UAAU;AACxN,WAAK,OAAO,SAAS,aAAa;AAClC,YAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IACrC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,kBAAkB,EAAE,QAAQ,6DAA6D,EAAE,UAAU,CAAC,WAAW,OAAO,SAAS,KAAK,OAAO,SAAS,gBAAgB,EAAE,SAAS,OAAO,UAAU;AAC1O,WAAK,OAAO,SAAS,mBAAmB;AACxC,YAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IACrC,CAAC,CAAC;AAEF,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,eAAe,EAAE,QAAQ,0KAA0K,EAAE,UAAU,CAAC,WAAW,OAAO,SAAS,KAAK,OAAO,SAAS,aAAa,EAAE,SAAS,OAAO,UAAU;AACjV,WAAK,OAAO,SAAS,gBAAgB;AACrC,YAAM,KAAK,OAAO,aAAa,IAAI;AAAA,IACrC,CAAC,CAAC;AAEF,gBAAY,SAAS,MAAM;AAAA,MACzB,MAAM;AAAA,IACR,CAAC;AAED,gBAAY,SAAS,MAAM;AAAA,MACzB,MAAM;AAAA,IACR,CAAC;AACD,QAAI,sBAAsB,YAAY,SAAS,KAAK;AACpD,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,aAAa,EAAE,QAAQ,yBAAyB,EAAE,UAAU,CAAC,WAAW,OAAO,cAAc,aAAa,EAAE,QAAQ,YAAY;AAExK,UAAI,QAAQ,wDAAwD,GAAG;AAErE,YAAG;AACD,gBAAM,KAAK,OAAO,wBAAwB,IAAI;AAC9C,8BAAoB,YAAY;AAAA,QAClC,SAAO,GAAN;AACC,8BAAoB,YAAY,uCAAuC;AAAA,QACzE;AAAA,MACF;AAAA,IACF,CAAC,CAAC;AAGF,gBAAY,SAAS,MAAM;AAAA,MACzB,MAAM;AAAA,IACR,CAAC;AACD,QAAI,cAAc,YAAY,SAAS,KAAK;AAC5C,SAAK,uBAAuB,WAAW;AAGvC,gBAAY,SAAS,MAAM;AAAA,MACzB,MAAM;AAAA,IACR,CAAC;AACD,QAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,eAAe,EAAE,QAAQ,oKAAoK,EAAE,UAAU,CAAC,WAAW,OAAO,cAAc,eAAe,EAAE,QAAQ,YAAY;AAEvT,UAAI,QAAQ,0HAA0H,GAAG;AAEvI,cAAM,KAAK,OAAO,8BAA8B;AAAA,MAClD;AAAA,IACF,CAAC,CAAC;AAAA,EAEJ;AAAA,EACA,oBAAoB;AAClB,WAAO,cAAc,kBAAkB,KAAK,OAAO,SAAS,QAAQ,EAAE,QAAQ,KAAK,IAAI;AAAA,EACzF;AAAA,EAEA,uBAAuB,aAAa;AAClC,gBAAY,MAAM;AAClB,QAAG,KAAK,OAAO,SAAS,aAAa,SAAS,GAAG;AAE/C,kBAAY,SAAS,KAAK;AAAA,QACxB,MAAM;AAAA,MACR,CAAC;AACD,UAAI,OAAO,YAAY,SAAS,IAAI;AACpC,eAAS,eAAe,KAAK,OAAO,SAAS,cAAc;AACzD,aAAK,SAAS,MAAM;AAAA,UAClB,MAAM;AAAA,QACR,CAAC;AAAA,MACH;AAEA,UAAI,SAAS,QAAQ,WAAW,EAAE,QAAQ,oBAAoB,EAAE,QAAQ,yBAAyB,EAAE,UAAU,CAAC,WAAW,OAAO,cAAc,yBAAyB,EAAE,QAAQ,YAAY;AAE3L,oBAAY,MAAM;AAElB,oBAAY,SAAS,KAAK;AAAA,UACxB,MAAM;AAAA,QACR,CAAC;AACD,cAAM,KAAK,OAAO,mBAAmB;AAErC,aAAK,uBAAuB,WAAW;AAAA,MACzC,CAAC,CAAC;AAAA,IACJ,OAAK;AACH,kBAAY,SAAS,KAAK;AAAA,QACxB,MAAM;AAAA,MACR,CAAC;AAAA,IACH;AAAA,EACF;AACF;AAEA,SAAS,gBAAgB,MAAM;AAC7B,SAAQ,KAAK,QAAQ,GAAG,MAAM,KAAO,CAAC,KAAK,GAAG,EAAE,QAAQ,KAAK,CAAC,CAAC,MAAM;AACvE;AAEA,IAAM,mCAAmC;AAEzC,IAAM,2BAAN,cAAuC,SAAS,SAAS;AAAA,EACvD,YAAY,MAAM,QAAQ;AACxB,UAAM,IAAI;AACV,SAAK,SAAS;AACd,SAAK,aAAa;AAClB,SAAK,gBAAgB;AACrB,SAAK,cAAc;AACnB,SAAK,OAAO;AACZ,SAAK,WAAW;AAChB,SAAK,iBAAiB;AACtB,SAAK,kBAAkB,CAAC;AACxB,SAAK,QAAQ,CAAC;AACd,SAAK,YAAY;AACjB,SAAK,oBAAoB;AACzB,SAAK,gBAAgB;AAAA,EACvB;AAAA,EACA,iBAAiB;AACf,WAAO;AAAA,EACT;AAAA,EACA,UAAU;AACR,WAAO;AAAA,EACT;AAAA,EACA,cAAc;AACZ,WAAO;AAAA,EACT;AAAA,EACA,SAAS;AACP,SAAK,SAAS;AACd,SAAK,OAAO,gBAAgB;AAAA,EAC9B;AAAA,EACA,UAAU;AACR,SAAK,KAAK,UAAU;AACpB,SAAK,IAAI,UAAU,0BAA0B,gCAAgC;AAAA,EAC/E;AAAA,EACA,cAAc;AACZ,SAAK,YAAY,MAAM;AACvB,SAAK,iBAAiB,KAAK,YAAY,UAAU,mBAAmB;AAEpE,SAAK,eAAe;AAEpB,SAAK,gBAAgB;AAErB,SAAK,kBAAkB;AACvB,SAAK,OAAO,aAAa,KAAK,aAAa,MAAM;AAAA,EACnD;AAAA;AAAA,EAEA,iBAAiB;AAEf,QAAI,oBAAoB,KAAK,eAAe,UAAU,sBAAsB;AAE5E,QAAI,YAAW,KAAK,KAAK,KAAK;AAC9B,QAAI,kBAAkB,kBAAkB,SAAS,SAAS;AAAA,MACxD,MAAM;AAAA,QACJ,MAAM;AAAA,QACN,OAAO;AAAA,MACT;AAAA,MACA,KAAK;AAAA,IACP,CAAC;AACD,oBAAgB,iBAAiB,UAAU,KAAK,YAAY,KAAK,IAAI,CAAC;AAGtE,QAAI,iBAAiB,KAAK,sBAAsB,mBAAmB,cAAc,mBAAmB;AACpG,mBAAe,iBAAiB,SAAS,KAAK,gBAAgB,KAAK,IAAI,CAAC;AAExE,QAAI,WAAW,KAAK,sBAAsB,mBAAmB,aAAa,MAAM;AAChF,aAAS,iBAAiB,SAAS,KAAK,UAAU,KAAK,IAAI,CAAC;AAE5D,QAAI,cAAc,KAAK,sBAAsB,mBAAmB,gBAAgB,SAAS;AACzF,gBAAY,iBAAiB,SAAS,KAAK,kBAAkB,KAAK,IAAI,CAAC;AAEvE,UAAM,eAAe,KAAK,sBAAsB,mBAAmB,YAAY,MAAM;AACrF,iBAAa,iBAAiB,SAAS,KAAK,SAAS,KAAK,IAAI,CAAC;AAAA,EACjE;AAAA,EACA,MAAM,oBAAoB;AACxB,UAAM,SAAS,MAAM,KAAK,IAAI,MAAM,QAAQ,KAAK,0BAA0B;AAC3E,SAAK,QAAQ,OAAO,MAAM,IAAI,CAAC,SAAS;AACtC,aAAO,KAAK,QAAQ,6BAA6B,EAAE,EAAE,QAAQ,SAAS,EAAE;AAAA,IAC1E,CAAC;AAED,QAAI,CAAC,KAAK;AACR,WAAK,QAAQ,IAAI,iCAAiC,KAAK,KAAK,IAAI;AAClE,SAAK,MAAM,KAAK;AAAA,EAClB;AAAA,EAEA,sBAAsB,mBAAmB,OAAO,OAAK,MAAM;AACzD,QAAI,MAAM,kBAAkB,SAAS,UAAU;AAAA,MAC7C,MAAM;AAAA,QACJ;AAAA,MACF;AAAA,IACF,CAAC;AACD,QAAG,MAAK;AACN,eAAS,QAAQ,KAAK,IAAI;AAAA,IAC5B,OAAK;AACH,UAAI,YAAY;AAAA,IAClB;AACA,WAAO;AAAA,EACT;AAAA;AAAA,EAEA,WAAW;AACT,SAAK,WAAW;AAChB,SAAK,YAAY;AAEjB,SAAK,oBAAoB,WAAW;AACpC,SAAK,WAAW,YAAY,QAAQ,kBAAkB,KAAK,OAAO,SAAS,QAAQ,EAAE,kBAAgB;AAAA,EACvG;AAAA;AAAA,EAEA,MAAM,UAAU,SAAS;AACvB,SAAK,WAAW;AAChB,UAAM,KAAK,KAAK,UAAU,OAAO;AACjC,SAAK,YAAY;AACjB,aAAS,IAAI,GAAG,IAAI,KAAK,KAAK,QAAQ,QAAQ,KAAK;AACjD,YAAM,KAAK,eAAe,KAAK,KAAK,QAAQ,CAAC,EAAE,SAAS,KAAK,KAAK,QAAQ,CAAC,EAAE,IAAI;AAAA,IACnF;AAAA,EACF;AAAA;AAAA,EAEA,aAAa;AACX,QAAI,KAAK,MAAM;AACb,WAAK,KAAK,UAAU;AAAA,IACtB;AACA,SAAK,OAAO,IAAI,0BAA0B,KAAK,MAAM;AAErD,QAAI,KAAK,oBAAoB;AAC3B,oBAAc,KAAK,kBAAkB;AAAA,IACvC;AAEA,SAAK,kBAAkB,CAAC;AAExB,SAAK,WAAW;AAAA,EAClB;AAAA,EAEA,YAAY,OAAO;AACjB,QAAI,gBAAgB,MAAM,OAAO;AACjC,SAAK,KAAK,YAAY,aAAa;AAAA,EACrC;AAAA;AAAA,EAGA,YAAY;AACV,SAAK,KAAK,UAAU;AACpB,QAAI,SAAS,OAAO,gCAAgC;AAAA,EACtD;AAAA,EAEA,kBAAkB;AAChB,SAAK,OAAO,UAAU;AAAA,EACxB;AAAA;AAAA,EAEA,kBAAkB;AAEhB,SAAK,WAAW,KAAK,eAAe,UAAU,aAAa;AAE3D,SAAK,oBAAoB,KAAK,SAAS,UAAU,sBAAsB;AAAA,EACzE;AAAA;AAAA,EAEA,6BAA6B;AAE3B,QAAG,CAAC,KAAK;AAAe,WAAK,gBAAgB,IAAI,gCAAgC,KAAK,KAAK,IAAI;AAC/F,SAAK,cAAc,KAAK;AAAA,EAC1B;AAAA;AAAA,EAEA,MAAM,+BAA+B;AAEnC,QAAG,CAAC,KAAK,iBAAgB;AACvB,WAAK,kBAAkB,IAAI,kCAAkC,KAAK,KAAK,IAAI;AAAA,IAC7E;AACA,SAAK,gBAAgB,KAAK;AAAA,EAC5B;AAAA;AAAA,EAEA,iBAAiB,aAAa;AAE5B,QAAI,YAAY,KAAK,SAAS;AAE9B,QAAI,cAAc,KAAK,SAAS,MAAM,UAAU,GAAG,SAAS;AAE5D,QAAI,aAAa,KAAK,SAAS,MAAM,UAAU,WAAW,KAAK,SAAS,MAAM,MAAM;AAEpF,SAAK,SAAS,QAAQ,cAAc,cAAc;AAElD,SAAK,SAAS,iBAAiB,YAAY,YAAY;AACvD,SAAK,SAAS,eAAe,YAAY,YAAY;AAErD,SAAK,SAAS,MAAM;AAAA,EACtB;AAAA;AAAA,EAGA,oBAAoB;AAElB,QAAI,aAAa,KAAK,eAAe,UAAU,cAAc;AAE7D,SAAK,WAAW,WAAW,SAAS,YAAY;AAAA,MAC9C,KAAK;AAAA,MACL,MAAM;AAAA,QACJ,aAAa,kBAAkB,KAAK,OAAO,SAAS,QAAQ,EAAE;AAAA,MAChE;AAAA,IACF,CAAC;AAID,eAAW,iBAAiB,SAAS,CAAC,MAAM;AAC1C,UAAG,CAAC,KAAK,GAAG,EAAE,QAAQ,EAAE,GAAG,MAAM;AAAI;AACrC,YAAM,YAAY,KAAK,SAAS;AAEhC,UAAI,EAAE,QAAQ,KAAK;AAEjB,YAAG,KAAK,SAAS,MAAM,YAAY,CAAC,MAAM,KAAI;AAE5C,eAAK,2BAA2B;AAChC;AAAA,QACF;AAAA,MACF,OAAK;AACH,aAAK,cAAc;AAAA,MACrB;AAEA,UAAI,EAAE,QAAQ,KAAK;AAGjB,YAAI,KAAK,SAAS,MAAM,WAAW,KAAK,KAAK,SAAS,MAAM,YAAY,CAAC,MAAM,KAAK;AAElF,eAAK,6BAA6B;AAClC;AAAA,QACF;AAAA,MACF;AAAA,IAEF,CAAC;AAED,eAAW,iBAAiB,WAAW,CAAC,MAAM;AAC5C,UAAI,EAAE,QAAQ,WAAW,EAAE,UAAU;AACnC,UAAE,eAAe;AACjB,YAAG,KAAK,eAAc;AACpB,kBAAQ,IAAI,yCAAyC;AACrD,cAAI,SAAS,OAAO,6DAA6D;AACjF;AAAA,QACF;AAEA,YAAI,aAAa,KAAK,SAAS;AAE/B,aAAK,SAAS,QAAQ;AAEtB,aAAK,oBAAoB,UAAU;AAAA,MACrC;AACA,WAAK,SAAS,MAAM,SAAS;AAC7B,WAAK,SAAS,MAAM,SAAU,KAAK,SAAS,eAAgB;AAAA,IAC9D,CAAC;AAED,QAAI,mBAAmB,WAAW,UAAU,qBAAqB;AAEjE,QAAI,eAAe,iBAAiB,SAAS,QAAQ,EAAE,MAAM,EAAC,IAAI,mBAAmB,OAAO,iBAAgB,EAAE,CAAC;AAC/G,aAAS,QAAQ,cAAc,QAAQ;AAEvC,iBAAa,iBAAiB,SAAS,MAAM;AAE3C,WAAK,WAAW;AAAA,IAClB,CAAC;AAED,QAAI,SAAS,iBAAiB,SAAS,UAAU,EAAE,MAAM,EAAC,IAAI,iBAAgB,GAAG,KAAK,cAAc,CAAC;AACrG,WAAO,YAAY;AAEnB,WAAO,iBAAiB,SAAS,MAAM;AACrC,UAAG,KAAK,eAAc;AACpB,gBAAQ,IAAI,yCAAyC;AACrD,YAAI,SAAS,OAAO,yCAAyC;AAC7D;AAAA,MACF;AAEA,UAAI,aAAa,KAAK,SAAS;AAE/B,WAAK,SAAS,QAAQ;AAEtB,WAAK,oBAAoB,UAAU;AAAA,IACrC,CAAC;AAAA,EACH;AAAA,EACA,MAAM,oBAAoB,YAAY;AACpC,SAAK,iBAAiB;AAEtB,UAAM,KAAK,eAAe,YAAY,MAAM;AAC5C,SAAK,KAAK,sBAAsB;AAAA,MAC9B,MAAM;AAAA,MACN,SAAS;AAAA,IACX,CAAC;AACD,UAAM,KAAK,iBAAiB;AAG5B,QAAG,KAAK,KAAK,uBAAuB,UAAU,GAAG;AAC/C,WAAK,KAAK,+BAA+B,YAAY,IAAI;AACzD;AAAA,IACF;AAQA,QAAG,KAAK,mCAAmC,UAAU,KAAK,KAAK,KAAK,0BAA0B,UAAU,GAAG;AAEzG,YAAM,UAAU,MAAM,KAAK,iBAAiB,UAAU;AAItD,YAAM,SAAS;AAAA,QACb;AAAA,UACE,MAAM;AAAA;AAAA,UAEN,SAAS;AAAA,QACX;AAAA,QACA;AAAA,UACE,MAAM;AAAA,UACN,SAAS;AAAA,QACX;AAAA,MACF;AACA,WAAK,2BAA2B,EAAC,UAAU,QAAQ,aAAa,EAAC,CAAC;AAClE;AAAA,IACF;AAEA,SAAK,2BAA2B;AAAA,EAClC;AAAA,EAEA,MAAM,mBAAmB;AACvB,QAAI,KAAK;AACP,oBAAc,KAAK,kBAAkB;AACvC,UAAM,KAAK,eAAe,OAAO,WAAW;AAE5C,QAAI,OAAO;AACX,SAAK,WAAW,YAAY;AAC5B,SAAK,qBAAqB,YAAY,MAAM;AAC1C;AACA,UAAI,OAAO;AACT,eAAO;AACT,WAAK,WAAW,YAAY,IAAI,OAAO,IAAI;AAAA,IAC7C,GAAG,GAAG;AAAA,EAGR;AAAA,EAEA,mBAAmB;AACjB,SAAK,gBAAgB;AAErB,QAAG,SAAS,eAAe,gBAAgB;AACzC,eAAS,eAAe,gBAAgB,EAAE,MAAM,UAAU;AAE5D,QAAG,SAAS,eAAe,iBAAiB;AAC1C,eAAS,eAAe,iBAAiB,EAAE,MAAM,UAAU;AAAA,EAC/D;AAAA,EACA,qBAAqB;AACnB,SAAK,gBAAgB;AAErB,QAAG,SAAS,eAAe,gBAAgB;AACzC,eAAS,eAAe,gBAAgB,EAAE,MAAM,UAAU;AAE5D,QAAG,SAAS,eAAe,iBAAiB;AAC1C,eAAS,eAAe,iBAAiB,EAAE,MAAM,UAAU;AAAA,EAC/D;AAAA;AAAA,EAIA,mCAAmC,YAAY;AAC7C,UAAM,UAAU,WAAW,MAAM,KAAK,OAAO,iBAAiB;AAC9D,QAAG;AAAS,aAAO;AACnB,WAAO;AAAA,EACT;AAAA;AAAA,EAGA,MAAM,eAAe,SAAS,OAAK,aAAa,cAAY,OAAO;AAEjE,QAAG,KAAK,oBAAoB;AAC1B,oBAAc,KAAK,kBAAkB;AACrC,WAAK,qBAAqB;AAE1B,WAAK,WAAW,YAAY;AAAA,IAC9B;AACA,QAAG,aAAa;AACd,WAAK,uBAAuB;AAC5B,UAAG,QAAQ,QAAQ,IAAI,MAAM,IAAI;AAC/B,aAAK,WAAW,aAAa;AAAA,MAC/B,OAAK;AACH,aAAK,WAAW,YAAY;AAE5B,cAAM,SAAS,iBAAiB,eAAe,KAAK,qBAAqB,KAAK,YAAY,gBAAgB,IAAI,SAAS,UAAU,CAAC;AAAA,MACpI;AAAA,IACF,OAAK;AACH,WAAK,sBAAsB;AAC3B,UAAI,KAAK,KAAK,OAAO,WAAW,KAAO,KAAK,cAAc,MAAO;AAE/D,aAAK,oBAAoB,IAAI;AAAA,MAC/B;AAEA,WAAK,WAAW,YAAY;AAC5B,YAAM,SAAS,iBAAiB,eAAe,SAAS,KAAK,YAAY,gBAAgB,IAAI,SAAS,UAAU,CAAC;AAEjH,WAAK,wBAAwB;AAE7B,WAAK,8BAA8B,OAAO;AAAA,IAC5C;AAEA,SAAK,kBAAkB,YAAY,KAAK,kBAAkB;AAAA,EAC5D;AAAA,EACA,8BAA8B,SAAS;AACrC,QAAI,KAAK,KAAK,WAAW,KAAK,KAAK,KAAK;AAEtC,YAAM,eAAe,KAAK,WAAW,SAAS,QAAQ;AAAA,QACpD,KAAK;AAAA,QACL,MAAM;AAAA,UACJ,OAAO;AAAA;AAAA,QACT;AAAA,MACF,CAAC;AACD,YAAM,WAAW,KAAK,KAAK;AAC3B,eAAS,QAAQ,cAAc,KAAK;AACpC,mBAAa,iBAAiB,SAAS,MAAM;AAE3C,kBAAU,UAAU,UAAU,2BAA2B,WAAW,SAAS;AAC7E,YAAI,SAAS,OAAO,4DAA4D;AAAA,MAClF,CAAC;AAAA,IACH;AACA,QAAG,KAAK,KAAK,SAAS;AAEpB,YAAM,qBAAqB,KAAK,WAAW,SAAS,QAAQ;AAAA,QAC1D,KAAK;AAAA,QACL,MAAM;AAAA,UACJ,OAAO;AAAA;AAAA,QACT;AAAA,MACF,CAAC;AACD,YAAM,eAAe,KAAK,KAAK,QAAQ,QAAQ,WAAW,MAAO,EAAE,SAAS;AAC5E,eAAS,QAAQ,oBAAoB,OAAO;AAC5C,yBAAmB,iBAAiB,SAAS,MAAM;AAEjD,kBAAU,UAAU,UAAU,wBAAwB,eAAe,SAAS;AAC9E,YAAI,SAAS,OAAO,iDAAiD;AAAA,MACvE,CAAC;AAAA,IACH;AAEA,UAAM,cAAc,KAAK,WAAW,SAAS,QAAQ;AAAA,MACnD,KAAK;AAAA,MACL,MAAM;AAAA,QACJ,OAAO;AAAA;AAAA,MACT;AAAA,IACF,CAAC;AACD,aAAS,QAAQ,aAAa,MAAM;AACpC,gBAAY,iBAAiB,SAAS,MAAM;AAE1C,gBAAU,UAAU,UAAU,QAAQ,SAAS,CAAC;AAChD,UAAI,SAAS,OAAO,iDAAiD;AAAA,IACvE,CAAC;AAAA,EACH;AAAA,EAEA,0BAA0B;AACxB,UAAM,QAAQ,KAAK,WAAW,iBAAiB,GAAG;AAElD,QAAI,MAAM,SAAS,GAAG;AACpB,eAAS,IAAI,GAAG,IAAI,MAAM,QAAQ,KAAK;AACrC,cAAM,OAAO,MAAM,CAAC;AACpB,cAAM,YAAY,KAAK,aAAa,WAAW;AAE/C,aAAK,iBAAiB,aAAa,CAAC,UAAU;AAC5C,eAAK,IAAI,UAAU,QAAQ,cAAc;AAAA,YACvC;AAAA,YACA,QAAQ;AAAA,YACR,aAAa,KAAK;AAAA,YAClB,UAAU;AAAA;AAAA,YAEV,UAAU;AAAA,UACZ,CAAC;AAAA,QACH,CAAC;AAED,aAAK,iBAAiB,SAAS,CAAC,UAAU;AACxC,gBAAM,aAAa,KAAK,IAAI,cAAc,qBAAqB,WAAW,GAAG;AAE7E,gBAAM,MAAM,SAAS,OAAO,WAAW,KAAK;AAE5C,cAAI,OAAO,KAAK,IAAI,UAAU,QAAQ,GAAG;AACzC,eAAK,SAAS,UAAU;AAAA,QAC1B,CAAC;AAAA,MACH;AAAA,IACF;AAAA,EACF;AAAA,EAEA,oBAAoB,MAAM;AACxB,QAAI,aAAa,KAAK,kBAAkB,UAAU,cAAc,MAAM;AAEtE,SAAK,aAAa,WAAW,UAAU,oBAAoB;AAE3D,SAAK,YAAY;AAAA,EACnB;AAAA,EAEA,MAAM,2BAA2B,OAAK,CAAC,GAAG;AACxC,UAAM,UAAU,KAAK,YAAY,KAAK,WAAW,KAAK,KAAK,gBAAgB;AAC3E,YAAQ,IAAI,WAAW,OAAO;AAC9B,UAAM,mBAAmB,KAAK,MAAM,cAAc,KAAK,OAAO,SAAS,gBAAgB,IAAI,CAAC;AAC5F,YAAQ,IAAI,oBAAoB,gBAAgB;AAChD,UAAM,iBAAiB,KAAK,MAAM,KAAK,UAAU,OAAO,EAAE,SAAS,CAAC;AACpE,YAAQ,IAAI,kBAAkB,cAAc;AAC5C,QAAI,uBAAuB,mBAAmB;AAE9C,QAAG,uBAAuB;AAAG,6BAAuB;AAAA,aAC5C,uBAAuB;AAAM,6BAAuB;AAC5D,YAAQ,IAAI,wBAAwB,oBAAoB;AACxD,WAAO;AAAA,MACL,OAAO,KAAK,OAAO,SAAS;AAAA,MAC5B,UAAU;AAAA;AAAA,MAEV,YAAY;AAAA,MACZ,aAAa;AAAA,MACb,OAAO;AAAA,MACP,kBAAkB;AAAA,MAClB,mBAAmB;AAAA,MACnB,QAAQ;AAAA,MACR,MAAM;AAAA,MACN,GAAG;AAAA;AAAA,MAEH,GAAG;AAAA,IACL;AAEA,QAAG,KAAK,QAAQ;AACd,YAAM,WAAW,MAAM,IAAI,QAAQ,CAAC,SAAS,WAAW;AACtD,YAAI;AAEF,gBAAM,MAAM;AACZ,eAAK,gBAAgB,IAAI,WAAW,KAAK;AAAA,YACvC,SAAS;AAAA,cACP,gBAAgB;AAAA,cAChB,eAAe,UAAU,KAAK,OAAO,SAAS;AAAA,YAChD;AAAA,YACA,QAAQ;AAAA,YACR,SAAS,KAAK,UAAU,IAAI;AAAA,UAC9B,CAAC;AACD,cAAI,MAAM;AACV,eAAK,cAAc,iBAAiB,WAAW,CAAC,MAAM;AACpD,gBAAI,EAAE,QAAQ,UAAU;AACtB,oBAAM,UAAU,KAAK,MAAM,EAAE,IAAI;AACjC,oBAAM,OAAO,QAAQ,QAAQ,CAAC,EAAE,MAAM;AACtC,kBAAI,CAAC,MAAM;AACT;AAAA,cACF;AACA,qBAAO;AACP,mBAAK,eAAe,MAAM,aAAa,IAAI;AAAA,YAC7C,OAAO;AACL,mBAAK,WAAW;AAChB,sBAAQ,GAAG;AAAA,YACb;AAAA,UACF,CAAC;AACD,eAAK,cAAc,iBAAiB,oBAAoB,CAAC,MAAM;AAC7D,gBAAI,EAAE,cAAc,GAAG;AACrB,sBAAQ,IAAI,iBAAiB,EAAE,UAAU;AAAA,YAC3C;AAAA,UACF,CAAC;AACD,eAAK,cAAc,iBAAiB,SAAS,CAAC,MAAM;AAClD,oBAAQ,MAAM,CAAC;AACf,gBAAI,SAAS,OAAO,sEAAsE;AAC1F,iBAAK,eAAe,8CAA8C,WAAW;AAC7E,iBAAK,WAAW;AAChB,mBAAO,CAAC;AAAA,UACV,CAAC;AACD,eAAK,cAAc,OAAO;AAAA,QAC5B,SAAS,KAAP;AACA,kBAAQ,MAAM,GAAG;AACjB,cAAI,SAAS,OAAO,sEAAsE;AAC1F,eAAK,WAAW;AAChB,iBAAO,GAAG;AAAA,QACZ;AAAA,MACF,CAAC;AAED,YAAM,KAAK,eAAe,UAAU,WAAW;AAC/C,WAAK,KAAK,sBAAsB;AAAA,QAC9B,MAAM;AAAA,QACN,SAAS;AAAA,MACX,CAAC;AACD;AAAA,IACF,OAAK;AACH,UAAG;AACD,cAAM,WAAW,OAAO,GAAG,SAAS,YAAY;AAAA,UAC9C,KAAK;AAAA,UACL,QAAQ;AAAA,UACR,SAAS;AAAA,YACP,eAAe,UAAU,KAAK,OAAO,SAAS;AAAA,YAC9C,gBAAgB;AAAA,UAClB;AAAA,UACA,aAAa;AAAA,UACb,MAAM,KAAK,UAAU,IAAI;AAAA,UACzB,OAAO;AAAA,QACT,CAAC;AAED,eAAO,KAAK,MAAM,SAAS,IAAI,EAAE,QAAQ,CAAC,EAAE,QAAQ;AAAA,MACtD,SAAO,KAAN;AACC,YAAI,SAAS,OAAO,kCAAkC,KAAK;AAAA,MAC7D;AAAA,IACF;AAAA,EACF;AAAA,EAEA,aAAa;AACX,QAAG,KAAK,eAAc;AACpB,WAAK,cAAc,MAAM;AACzB,WAAK,gBAAgB;AAAA,IACvB;AACA,SAAK,mBAAmB;AACxB,QAAG,KAAK,oBAAmB;AACzB,oBAAc,KAAK,kBAAkB;AACrC,WAAK,qBAAqB;AAE1B,WAAK,WAAW,cAAc,OAAO;AACrC,WAAK,aAAa;AAAA,IACpB;AAAA,EACF;AAAA,EAEA,MAAM,iBAAiB,YAAY;AACjC,SAAK,KAAK,cAAc;AAExB,UAAM,YAAY;AAElB,UAAM,SAAS;AAAA,MACb;AAAA,QACE,MAAM;AAAA,QACN,SAAS;AAAA,MACX;AAAA,MACA;AAAA,QACE,MAAM;AAAA,QACN,SAAS;AAAA,MACX;AAAA,IACF;AACA,UAAM,MAAM,MAAM,KAAK,2BAA2B;AAAA,MAChD,UAAU;AAAA,MACV,QAAQ;AAAA,MACR,aAAa;AAAA,MACb,YAAY;AAAA,IACd,CAAC;AACD,SAAK,KAAK,MAAM;AAEhB,QAAI,SAAS,CAAC;AAEd,QAAG,KAAK,KAAK,0BAA0B,UAAU,GAAG;AAElD,YAAM,cAAc,KAAK,KAAK,sBAAsB,UAAU;AAG9D,UAAG,aAAY;AACb,iBAAS;AAAA,UACP,kBAAkB;AAAA,QACpB;AAAA,MACF;AAAA,IACF;AAEA,QAAI,UAAU,MAAM,KAAK,OAAO,IAAI,OAAO,KAAK,MAAM;AACtD,YAAQ,IAAI,WAAW,QAAQ,MAAM;AACrC,cAAU,KAAK,2CAA2C,OAAO;AACjE,YAAQ,IAAI,+BAA+B,QAAQ,MAAM;AACzD,cAAU,KAAK,gCAAgC,OAAO;AAEtD,WAAO,MAAM,KAAK,uBAAuB,OAAO;AAAA,EAClD;AAAA,EAGA,gCAAgC,SAAS;AAEvC,cAAU,QAAQ,KAAK,CAAC,GAAG,MAAM;AAC/B,YAAM,UAAU,EAAE,aAAa,EAAE;AACjC,YAAM,UAAU,EAAE,aAAa,EAAE;AAEjC,UAAI,UAAU;AACZ,eAAO;AAET,UAAI,UAAU;AACZ,eAAO;AAET,aAAO;AAAA,IACT,CAAC;AACD,WAAO;AAAA,EACT;AAAA,EAEA,2CAA2C,SAAS;AAElD,UAAM,MAAM,QAAQ,IAAI,CAAC,MAAM,EAAE,UAAU;AAC3C,UAAM,OAAO,IAAI,OAAO,CAAC,GAAG,MAAM,IAAI,CAAC,IAAI,IAAI;AAC/C,QAAI,UAAU,KAAK,KAAK,IAAI,IAAI,CAAC,MAAM,KAAK,IAAI,IAAI,MAAM,CAAC,CAAC,EAAE,OAAO,CAAC,GAAG,MAAM,IAAI,CAAC,IAAI,IAAI,MAAM;AAElG,QAAI,UAAU;AACd,WAAO,UAAU,QAAQ,QAAQ;AAC/B,YAAM,OAAO,QAAQ,UAAU,CAAC;AAChC,UAAI,MAAM;AACR,cAAM,WAAW,KAAK,IAAI,KAAK,aAAa,QAAQ,OAAO,EAAE,UAAU;AACvE,YAAI,WAAW,SAAS;AACtB,cAAG,UAAU;AAAG,sBAAU,UAAU;AAAA;AAC/B;AAAA,QACP;AAAA,MACF;AACA;AAAA,IACF;AAEA,cAAU,QAAQ,MAAM,GAAG,UAAQ,CAAC;AACpC,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAWA,MAAM,uBAAuB,SAAS;AACpC,QAAI,UAAU,CAAC;AACf,UAAM,cAAe,KAAK,OAAO,SAAS,qBAAqB,uBAAwB,KAAK;AAC5F,UAAM,YAAY,cAAc,KAAK,OAAO,SAAS,gBAAgB,IAAI;AACzE,QAAI,aAAa;AACjB,aAAS,IAAI,GAAG,IAAI,QAAQ,QAAQ,KAAK;AACvC,UAAI,QAAQ,UAAU;AACpB;AACF,UAAI,cAAc;AAChB;AACF,UAAI,OAAO,QAAQ,CAAC,EAAE,SAAS;AAC7B;AAEF,YAAM,cAAc,QAAQ,CAAC,EAAE,KAAK,QAAQ,MAAM,KAAK,EAAE,QAAQ,OAAO,EAAE,EAAE,QAAQ,OAAO,KAAK;AAChG,UAAI,cAAc,GAAG;AAAA;AAErB,YAAM,sBAAsB,YAAY,aAAa,YAAY;AACjE,UAAI,QAAQ,CAAC,EAAE,KAAK,QAAQ,GAAG,MAAM,IAAI;AACvC,uBAAe,MAAM,KAAK,OAAO,gBAAgB,QAAQ,CAAC,EAAE,MAAM,EAAE,WAAW,oBAAoB,CAAC;AAAA,MACtG,OAAO;AACL,uBAAe,MAAM,KAAK,OAAO,eAAe,QAAQ,CAAC,EAAE,MAAM,EAAE,WAAW,oBAAoB,CAAC;AAAA,MACrG;AAEA,oBAAc,YAAY;AAE1B,cAAQ,KAAK;AAAA,QACX,MAAM,QAAQ,CAAC,EAAE;AAAA,QACjB,MAAM;AAAA,MACR,CAAC;AAAA,IACH;AAEA,YAAQ,IAAI,sBAAsB,QAAQ,MAAM;AAEhD,YAAQ,IAAI,4BAA4B,KAAK,MAAM,aAAa,GAAG,CAAC;AAEpE,SAAK,KAAK,UAAU,4EAA4E,QAAQ,wIAAwI,kBAAkB,KAAK,OAAO,SAAS,QAAQ,EAAE;AACjS,aAAQ,IAAI,GAAG,IAAI,QAAQ,QAAQ,KAAK;AACtC,WAAK,KAAK,WAAW;AAAA,YAAe,IAAE;AAAA,EAAS,QAAQ,CAAC,EAAE;AAAA,UAAiB,IAAE;AAAA,IAC/E;AACA,WAAO,KAAK,KAAK;AAAA,EACnB;AAGF;AAEA,SAAS,cAAc,QAAM,iBAAiB;AAC5C,QAAM,eAAe;AAAA,IACnB,qBAAqB;AAAA,IACrB,SAAS;AAAA,IACT,iBAAiB;AAAA,IACjB,sBAAsB;AAAA,EACxB;AACA,SAAO,aAAa,KAAK;AAC3B;AAaA,IAAM,4BAAN,MAAgC;AAAA,EAC9B,YAAY,QAAQ;AAClB,SAAK,MAAM,OAAO;AAClB,SAAK,SAAS;AACd,SAAK,UAAU;AACf,SAAK,UAAU,CAAC;AAChB,SAAK,UAAU;AACf,SAAK,MAAM;AACX,SAAK,SAAS,CAAC;AAAA,EACjB;AAAA,EACA,MAAM,YAAY;AAEhB,QAAI,KAAK,OAAO,WAAW;AAAG;AAG9B,QAAI,CAAE,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,0BAA0B,GAAI;AACtE,YAAM,KAAK,IAAI,MAAM,QAAQ,MAAM,0BAA0B;AAAA,IAC/D;AAEA,QAAI,CAAC,KAAK,SAAS;AACjB,WAAK,UAAU,KAAK,KAAK,IAAI,WAAM,KAAK,qBAAqB;AAAA,IAC/D;AAEA,QAAI,CAAC,KAAK,QAAQ,MAAM,qBAAqB,GAAG;AAC9C,cAAQ,IAAI,sBAAsB,KAAK,OAAO;AAC9C,UAAI,SAAS,OAAO,gEAAgE,KAAK,UAAU,GAAG;AAAA,IACxG;AAEA,UAAM,YAAY,KAAK,UAAU;AACjC,SAAK,IAAI,MAAM,QAAQ;AAAA,MACrB,8BAA8B;AAAA,MAC9B,KAAK,UAAU,KAAK,QAAQ,MAAM,CAAC;AAAA,IACrC;AAAA,EACF;AAAA,EACA,MAAM,UAAU,SAAS;AACvB,SAAK,UAAU;AAGf,UAAM,YAAY,KAAK,UAAU;AAEjC,QAAI,YAAY,MAAM,KAAK,IAAI,MAAM,QAAQ;AAAA,MAC3C,8BAA8B;AAAA,IAChC;AAEA,SAAK,SAAS,KAAK,MAAM,SAAS;AAElC,SAAK,UAAU,KAAK,gBAAgB;AAAA,EAKtC;AAAA;AAAA;AAAA,EAGA,gBAAgB,yBAAuB,CAAC,GAAG;AAEzC,QAAG,uBAAuB,WAAW,GAAE;AACrC,WAAK,UAAU,KAAK,OAAO,IAAI,UAAQ;AACrC,eAAO,KAAK,KAAK,SAAS,CAAC;AAAA,MAC7B,CAAC;AAAA,IACH,OAAK;AAGH,UAAI,uBAAuB,CAAC;AAC5B,eAAQ,IAAI,GAAG,IAAI,uBAAuB,QAAQ,KAAI;AACpD,6BAAqB,uBAAuB,CAAC,EAAE,CAAC,CAAC,IAAI,uBAAuB,CAAC,EAAE,CAAC;AAAA,MAClF;AAEA,WAAK,UAAU,KAAK,OAAO,IAAI,CAAC,MAAM,eAAe;AAEnD,YAAG,qBAAqB,UAAU,MAAM,QAAU;AAChD,iBAAO,KAAK,qBAAqB,UAAU,CAAC;AAAA,QAC9C;AAEA,eAAO,KAAK,KAAK,SAAS,CAAC;AAAA,MAC7B,CAAC;AAAA,IACH;AAEA,SAAK,UAAU,KAAK,QAAQ,IAAI,aAAW;AACzC,aAAO;AAAA,QACL,MAAM,QAAQ;AAAA,QACd,SAAS,QAAQ;AAAA,MACnB;AAAA,IACF,CAAC;AACD,WAAO,KAAK;AAAA,EACd;AAAA,EACA,OAAO;AAEL,WAAO,KAAK,OAAO,KAAK,OAAO,SAAS,CAAC,EAAE,KAAK,OAAO,KAAK,OAAO,SAAS,CAAC,EAAE,SAAS,CAAC;AAAA,EAC3F;AAAA,EACA,YAAY;AACV,WAAO,KAAK,KAAK,EAAE;AAAA,EACrB;AAAA;AAAA,EAEA,eAAe;AACb,WAAO,KAAK,KAAK,EAAE;AAAA,EACrB;AAAA;AAAA;AAAA,EAGA,sBAAsB,SAAS,OAAK,IAAI;AAEtC,QAAG,KAAK,SAAQ;AACd,cAAQ,UAAU,KAAK;AACvB,WAAK,UAAU;AAAA,IACjB;AACA,QAAG,KAAK,KAAI;AACV,cAAQ,MAAM,KAAK;AACnB,WAAK,MAAM;AAAA,IACb;AACA,QAAI,SAAS,IAAI;AACf,WAAK,OAAO,KAAK,CAAC,OAAO,CAAC;AAAA,IAC5B,OAAK;AAEH,WAAK,OAAO,IAAI,EAAE,KAAK,OAAO;AAAA,IAChC;AAAA,EACF;AAAA,EACA,gBAAe;AACb,SAAK,UAAU;AACf,SAAK,MAAM;AAAA,EACb;AAAA,EACA,MAAM,YAAY,UAAS;AAEzB,QAAI,KAAK,WAAW,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,8BAA8B,KAAK,UAAU,OAAO,GAAG;AAC7G,iBAAW,KAAK,QAAQ,QAAQ,KAAK,KAAK,GAAG,QAAQ;AAErD,YAAM,KAAK,IAAI,MAAM,QAAQ;AAAA,QAC3B,8BAA8B,KAAK,UAAU;AAAA,QAC7C,8BAA8B,WAAW;AAAA,MAC3C;AAEA,WAAK,UAAU;AAAA,IACjB,OAAK;AACH,WAAK,UAAU,WAAW,WAAM,KAAK,qBAAqB;AAE1D,YAAM,KAAK,UAAU;AAAA,IACvB;AAAA,EAEF;AAAA,EAEA,OAAO;AACL,QAAG,KAAK,SAAQ;AAEd,aAAO,KAAK,QAAQ,QAAQ,WAAU,EAAE;AAAA,IAC1C;AACA,WAAO;AAAA,EACT;AAAA,EAEA,uBAAuB;AACrB,YAAO,oBAAI,KAAK,GAAE,YAAY,EAAE,QAAQ,eAAe,GAAG,EAAE,KAAK;AAAA,EACnE;AAAA;AAAA,EAEA,MAAM,+BAA+B,YAAY,WAAW;AAC1D,QAAI,eAAe;AAEnB,UAAM,QAAQ,KAAK,uBAAuB,UAAU;AAEpD,QAAI,YAAY,cAAc,KAAK,OAAO,SAAS,gBAAgB;AACnE,aAAQ,IAAI,GAAG,IAAI,MAAM,QAAQ,KAAI;AAEnC,YAAM,iBAAkB,MAAM,SAAS,IAAI,IAAK,KAAK,MAAM,aAAa,MAAM,SAAS,EAAE,IAAI;AAE7F,YAAM,eAAe,MAAM,KAAK,kBAAkB,MAAM,CAAC,GAAG,EAAC,YAAY,eAAc,CAAC;AACxF,cAAQ,IAAI,YAAY;AACxB,sBAAgB,oBAAoB,MAAM,CAAC,EAAE;AAAA;AAC7C,sBAAgB;AAChB,sBAAgB;AAAA;AAChB,mBAAa,aAAa;AAC1B,UAAG,aAAa;AAAG;AAAA,IACrB;AACA,SAAK,UAAU;AACf,UAAM,SAAS;AAAA,MACb;AAAA,QACE,MAAM;AAAA,QACN,SAAS;AAAA,MACX;AAAA,MACA;AAAA,QACE,MAAM;AAAA,QACN,SAAS;AAAA,MACX;AAAA,IACF;AACA,cAAU,2BAA2B,EAAC,UAAU,QAAQ,aAAa,EAAC,CAAC;AAAA,EACzE;AAAA;AAAA,EAEA,uBAAuB,YAAY;AACjC,QAAG,WAAW,QAAQ,IAAI,MAAM;AAAI,aAAO;AAC3C,QAAG,WAAW,QAAQ,IAAI,MAAM;AAAI,aAAO;AAC3C,WAAO;AAAA,EACT;AAAA;AAAA,EAEA,0BAA0B,YAAY;AACpC,QAAG,WAAW,QAAQ,GAAG,MAAM;AAAI,aAAO;AAC1C,QAAG,WAAW,QAAQ,GAAG,MAAM,WAAW,YAAY,GAAG;AAAG,aAAO;AACnE,WAAO;AAAA,EACT;AAAA;AAAA,EAEA,sBAAsB,YAAY;AAEhC,UAAM,UAAU,KAAK,OAAO,QAAQ,MAAM;AAC1C,UAAM,UAAU,QAAQ,KAAK,CAAC,GAAG,MAAM,EAAE,SAAS,EAAE,MAAM,EAAE,IAAI,YAAU;AAExE,UAAG,WAAW,QAAQ,MAAM,MAAM,IAAG;AAEnC,qBAAa,WAAW,QAAQ,QAAQ,EAAE;AAC1C,eAAO;AAAA,MACT;AACA,aAAO;AAAA,IACT,CAAC,EAAE,OAAO,YAAU,MAAM;AAC1B,YAAQ,IAAI,OAAO;AAEnB,QAAG;AAAS,aAAO;AACnB,WAAO;AAAA,EACT;AAAA;AAAA,EAIA,uBAAuB,YAAY;AACjC,UAAM,UAAU,WAAW,MAAM,gBAAgB;AACjD,YAAQ,IAAI,OAAO;AAEnB,QAAG;AAAS,aAAO,QAAQ,IAAI,WAAS;AACtC,eAAO,KAAK,IAAI,cAAc,qBAAqB,MAAM,QAAQ,MAAM,EAAE,EAAE,QAAQ,MAAM,EAAE,GAAG,GAAG;AAAA,MACnG,CAAC;AACD,WAAO,CAAC;AAAA,EACV;AAAA;AAAA,EAEA,MAAM,kBAAkB,MAAM,OAAK,CAAC,GAAG;AACrC,WAAO;AAAA,MACL,YAAY;AAAA,MACZ,GAAG;AAAA,IACL;AAEA,QAAG,EAAE,gBAAgB,SAAS;AAAQ,aAAO;AAE7C,QAAI,eAAe,MAAM,KAAK,IAAI,MAAM,WAAW,IAAI;AAEvD,QAAI,KAAK,OAAO,SAAS,qBAAqB;AAC5C,qBAAe,aAAa,QAAQ,qBAAoB,EAAE;AAAA,IAC5D;AAEA,QAAG,aAAa,QAAQ,aAAa,IAAI,IAAG;AAE1C,qBAAe,MAAM,KAAK,wBAAwB,cAAc,KAAK,MAAM,IAAI;AAAA,IACjF;AACA,WAAO,aAAa,UAAU,GAAG,KAAK,UAAU;AAAA,EAClD;AAAA,EAGA,MAAM,wBAAwB,cAAc,WAAW,OAAK,CAAC,GAAG;AAC9D,WAAO;AAAA,MACL,YAAY;AAAA,MACZ,GAAG;AAAA,IACL;AAEA,UAAM,eAAe,OAAO,aAAa;AAEzC,QAAG,CAAC;AAAc,aAAO;AACzB,UAAM,uBAAuB,aAAa,MAAM,uBAAuB;AAEvE,aAAS,IAAI,GAAG,IAAI,qBAAqB,QAAQ,KAAK;AAEpD,UAAG,KAAK,cAAc,KAAK,aAAa,aAAa,QAAQ,qBAAqB,CAAC,CAAC;AAAG;AAEvF,YAAM,sBAAsB,qBAAqB,CAAC;AAElD,YAAM,8BAA8B,oBAAoB,QAAQ,eAAe,EAAE,EAAE,QAAQ,OAAO,EAAE;AAEpG,YAAM,wBAAwB,MAAM,aAAa,cAAc,6BAA6B,WAAW,IAAI;AAE3G,UAAI,sBAAsB,YAAY;AACpC,uBAAe,aAAa,QAAQ,qBAAqB,sBAAsB,KAAK;AAAA,MACtF;AAAA,IACF;AACA,WAAO;AAAA,EACT;AACF;AAEA,IAAM,mCAAN,cAA+C,SAAS,kBAAkB;AAAA,EACxE,YAAY,KAAK,MAAM,OAAO;AAC5B,UAAM,GAAG;AACT,SAAK,MAAM;AACX,SAAK,OAAO;AACZ,SAAK,eAAe,oCAAoC;AAAA,EAC1D;AAAA,EACA,WAAW;AACT,QAAI,CAAC,KAAK,KAAK,OAAO;AACpB,aAAO,CAAC;AAAA,IACV;AACA,WAAO,KAAK,KAAK;AAAA,EACnB;AAAA,EACA,YAAY,MAAM;AAEhB,QAAG,KAAK,QAAQ,UAAU,MAAM,IAAG;AACjC,WAAK,QAAQ,WAAU,EAAE;AAAA,IAC3B;AACA,WAAO;AAAA,EACT;AAAA,EACA,aAAa,SAAS;AACpB,SAAK,KAAK,UAAU,OAAO;AAAA,EAC7B;AACF;AAGA,IAAM,kCAAN,cAA8C,SAAS,kBAAkB;AAAA,EACvE,YAAY,KAAK,MAAM;AACrB,UAAM,GAAG;AACT,SAAK,MAAM;AACX,SAAK,OAAO;AACZ,SAAK,eAAe,4BAA4B;AAAA,EAClD;AAAA,EACA,WAAW;AAET,WAAO,KAAK,IAAI,MAAM,iBAAiB,EAAE,KAAK,CAAC,GAAG,MAAM,EAAE,SAAS,cAAc,EAAE,QAAQ,CAAC;AAAA,EAC9F;AAAA,EACA,YAAY,MAAM;AAChB,WAAO,KAAK;AAAA,EACd;AAAA,EACA,aAAa,MAAM;AACjB,SAAK,KAAK,iBAAiB,KAAK,WAAW,KAAK;AAAA,EAClD;AACF;AAEA,IAAM,oCAAN,cAAgD,SAAS,kBAAkB;AAAA,EACzE,YAAY,KAAK,MAAM;AACrB,UAAM,GAAG;AACT,SAAK,MAAM;AACX,SAAK,OAAO;AACZ,SAAK,eAAe,8BAA8B;AAAA,EACpD;AAAA,EACA,WAAW;AACT,WAAO,KAAK,KAAK,OAAO;AAAA,EAC1B;AAAA,EACA,YAAY,MAAM;AAChB,WAAO;AAAA,EACT;AAAA,EACA,aAAa,QAAQ;AACnB,SAAK,KAAK,iBAAiB,SAAS,IAAI;AAAA,EAC1C;AACF;AAIA,IAAM,aAAN,MAAiB;AAAA;AAAA,EAEf,YAAY,KAAK,SAAS;AAExB,cAAU,WAAW,CAAC;AACtB,SAAK,MAAM;AACX,SAAK,SAAS,QAAQ,UAAU;AAChC,SAAK,UAAU,QAAQ,WAAW,CAAC;AACnC,SAAK,UAAU,QAAQ,WAAW;AAClC,SAAK,kBAAkB,QAAQ,mBAAmB;AAClD,SAAK,YAAY,CAAC;AAClB,SAAK,aAAa,KAAK;AACvB,SAAK,WAAW;AAChB,SAAK,QAAQ;AACb,SAAK,MAAM;AACX,SAAK,kBAAkB;AACvB,SAAK,eAAe;AACpB,SAAK,aAAa;AAClB,SAAK,OAAO;AACZ,SAAK,SAAS;AAAA,EAChB;AAAA;AAAA,EAEA,iBAAiB,MAAM,UAAU;AAE/B,QAAI,CAAC,KAAK,UAAU,IAAI,GAAG;AACzB,WAAK,UAAU,IAAI,IAAI,CAAC;AAAA,IAC1B;AAEA,QAAG,KAAK,UAAU,IAAI,EAAE,QAAQ,QAAQ,MAAM,IAAI;AAChD,WAAK,UAAU,IAAI,EAAE,KAAK,QAAQ;AAAA,IACpC;AAAA,EACF;AAAA;AAAA,EAEA,oBAAoB,MAAM,UAAU;AAElC,QAAI,CAAC,KAAK,UAAU,IAAI,GAAG;AACzB;AAAA,IACF;AACA,QAAI,WAAW,CAAC;AAEhB,aAAS,IAAI,GAAG,IAAI,KAAK,UAAU,IAAI,EAAE,QAAQ,KAAK;AAEpD,UAAI,KAAK,UAAU,IAAI,EAAE,CAAC,MAAM,UAAU;AACxC,iBAAS,KAAK,KAAK,UAAU,IAAI,EAAE,CAAC,CAAC;AAAA,MACvC;AAAA,IACF;AAEA,QAAI,KAAK,UAAU,IAAI,EAAE,WAAW,GAAG;AACrC,aAAO,KAAK,UAAU,IAAI;AAAA,IAC5B,OAAO;AACL,WAAK,UAAU,IAAI,IAAI;AAAA,IACzB;AAAA,EACF;AAAA;AAAA,EAEA,cAAc,OAAO;AAEnB,QAAI,CAAC,OAAO;AACV,aAAO;AAAA,IACT;AAEA,UAAM,SAAS;AAEf,QAAI,YAAY,OAAO,MAAM;AAE7B,QAAI,KAAK,eAAe,SAAS,GAAG;AAElC,WAAK,SAAS,EAAE,KAAK,MAAM,KAAK;AAEhC,UAAI,MAAM,kBAAkB;AAC1B,eAAO;AAAA,MACT;AAAA,IACF;AAEA,QAAI,KAAK,UAAU,MAAM,IAAI,GAAG;AAC9B,aAAO,KAAK,UAAU,MAAM,IAAI,EAAE,MAAM,SAAS,UAAU;AACzD,iBAAS,KAAK;AACd,eAAO,CAAC,MAAM;AAAA,MAChB,CAAC;AAAA,IACH;AACA,WAAO;AAAA,EACT;AAAA;AAAA,EAEA,eAAe,OAAO;AAEpB,QAAI,QAAQ,IAAI,YAAY,kBAAkB;AAE9C,UAAM,aAAa;AAEnB,SAAK,aAAa;AAElB,SAAK,cAAc,KAAK;AAAA,EAC1B;AAAA;AAAA,EAEA,iBAAiB,GAAG;AAElB,QAAI,QAAQ,IAAI,YAAY,OAAO;AAEnC,UAAM,OAAO,EAAE,cAAc;AAE7B,SAAK,cAAc,KAAK;AACxB,SAAK,MAAM;AAAA,EACb;AAAA;AAAA,EAEA,eAAe,GAAG;AAEhB,QAAI,QAAQ,IAAI,YAAY,OAAO;AAEnC,SAAK,MAAM;AAAA,EACb;AAAA;AAAA,EAEA,kBAAkB,GAAG;AAEnB,QAAI,CAAC,KAAK,KAAK;AACb;AAAA,IACF;AAEA,QAAI,KAAK,IAAI,WAAW,KAAK;AAE3B,WAAK,iBAAiB,CAAC;AACvB;AAAA,IACF;AAEA,QAAI,KAAK,eAAe,KAAK,YAAY;AAEvC,WAAK,cAAc,IAAI,YAAY,MAAM,CAAC;AAE1C,WAAK,eAAe,KAAK,IAAI;AAAA,IAC/B;AAEA,QAAI,OAAO,KAAK,IAAI,aAAa,UAAU,KAAK,QAAQ;AAExD,SAAK,YAAY,KAAK;AAEtB,SAAK,MAAM,kBAAkB,EAAE,QAAQ,SAAS,MAAK;AACnD,UAAG,KAAK,KAAK,EAAE,WAAW,GAAG;AAC3B,aAAK,cAAc,KAAK,iBAAiB,KAAK,MAAM,KAAK,CAAC,CAAC;AAC3D,aAAK,QAAQ;AAAA,MACf,OAAO;AACL,aAAK,SAAS;AAAA,MAChB;AAAA,IACF,EAAE,KAAK,IAAI,CAAC;AAAA,EACd;AAAA;AAAA,EAEA,gBAAgB,GAAG;AACjB,SAAK,kBAAkB,CAAC;AAExB,SAAK,cAAc,KAAK,iBAAiB,KAAK,KAAK,CAAC;AACpD,SAAK,QAAQ;AAAA,EACf;AAAA;AAAA,EAEA,iBAAiB,OAAO;AAEtB,QAAI,CAAC,SAAS,MAAM,WAAW,GAAG;AAChC,aAAO;AAAA,IACT;AAEA,QAAI,IAAI,EAAC,IAAI,MAAM,OAAO,MAAM,MAAM,IAAI,OAAO,UAAS;AAE1D,UAAM,MAAM,cAAc,EAAE,QAAQ,SAAS,MAAM;AACjD,aAAO,KAAK,UAAU;AACtB,UAAI,QAAQ,KAAK,QAAQ,KAAK,eAAe;AAC7C,UAAG,SAAS,GAAG;AACb;AAAA,MACF;AAEA,UAAI,QAAQ,KAAK,UAAU,GAAG,KAAK;AACnC,UAAG,EAAE,SAAS,IAAI;AAChB;AAAA,MACF;AAEA,UAAI,QAAQ,KAAK,UAAU,QAAQ,CAAC,EAAE,SAAS;AAC/C,UAAG,UAAU,QAAQ;AACnB,UAAE,KAAK,KAAK;AAAA,MACd,OAAO;AACL,UAAE,KAAK,IAAI;AAAA,MACb;AAAA,IACF,EAAE,KAAK,IAAI,CAAC;AAEZ,QAAI,QAAQ,IAAI,YAAY,EAAE,KAAK;AACnC,UAAM,OAAO,EAAE;AACf,UAAM,KAAK,EAAE;AACb,WAAO;AAAA,EACT;AAAA;AAAA,EAEA,qBAAqB;AACnB,QAAG,CAAC,KAAK,KAAK;AACZ;AAAA,IACF;AACA,QAAG,KAAK,IAAI,eAAe,eAAe,MAAM;AAC9C,WAAK,eAAe,KAAK,MAAM;AAAA,IACjC;AAAA,EACF;AAAA;AAAA,EAEA,SAAS;AAEP,SAAK,eAAe,KAAK,UAAU;AAEnC,SAAK,MAAM,IAAI,eAAe;AAE9B,SAAK,IAAI,iBAAiB,YAAY,KAAK,kBAAkB,KAAK,IAAI,CAAC;AAEvE,SAAK,IAAI,iBAAiB,QAAQ,KAAK,gBAAgB,KAAK,IAAI,CAAC;AAEjE,SAAK,IAAI,iBAAiB,oBAAoB,KAAK,mBAAmB,KAAK,IAAI,CAAC;AAEhF,SAAK,IAAI,iBAAiB,SAAS,KAAK,iBAAiB,KAAK,IAAI,CAAC;AAEnE,SAAK,IAAI,iBAAiB,SAAS,KAAK,eAAe,KAAK,IAAI,CAAC;AAEjE,SAAK,IAAI,KAAK,KAAK,QAAQ,KAAK,GAAG;AAEnC,aAAS,UAAU,KAAK,SAAS;AAC/B,WAAK,IAAI,iBAAiB,QAAQ,KAAK,QAAQ,MAAM,CAAC;AAAA,IACxD;AAEA,SAAK,IAAI,kBAAkB,KAAK;AAEhC,SAAK,IAAI,KAAK,KAAK,OAAO;AAAA,EAC5B;AAAA;AAAA,EAEA,QAAQ;AACN,QAAG,KAAK,eAAe,KAAK,QAAQ;AAClC;AAAA,IACF;AACA,SAAK,IAAI,MAAM;AACf,SAAK,MAAM;AACX,SAAK,eAAe,KAAK,MAAM;AAAA,EACjC;AACF;AAEA,OAAO,UAAU;",
  "names": ["line_limit", "item", "link", "file_link", "file_link_list"]
}
 diff --git a/src/index.js b/src/index.js index a8ff3a08..c2e69efc 100644 --- a/src/index.js +++ b/src/index.js @@ -1832,7 +1832,6 @@ class SmartConnectionsPlugin extends Obsidian.Plugin { * BEGIN EXTERNAL LINK LOGIC * if link is an object, it indicates external link */ - console.log(this); if (typeof nearest[i].link === "object") { const item = list.createEl("div", { cls: "search-result" }); const link = item.createEl("a", { From c7855724887db881719f4796c5a19c9e2b64eb64 Mon Sep 17 00:00:00 2001 From: Mara-Li Date: Tue, 2 Jan 2024 13:20:11 +0100 Subject: [PATCH 09/11] Delete data.json --- data.json | 25 ------------------------- 1 file changed, 25 deletions(-) delete mode 100644 data.json diff --git a/data.json b/data.json deleted file mode 100644 index aa5ba945..00000000 --- a/data.json +++ /dev/null @@ -1,25 +0,0 @@ -{ - "api_key": "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", - "chat_open": false, - "file_exclusions": "sortspec", - "folder_exclusions": "", - "header_exclusions": "", - "path_only": "", - "show_full_path": false, - "cut_off_frontmatter": true, - "expanded_view": false, - "group_nearest_by_file": false, - "language": "fr", - "log_render": false, - "log_render_files": true, - "recently_sent_retry_notice": false, - "skip_sections": false, - "smart_chat_model": "gpt-4-1106-preview", - "view_open": true, - "version": "1.6.47", - "open_in_big_view": false, - "results_count": 30, - "failed_files": [ - "" - ] -} \ No newline at end of file From aebbcdbb8f1a08156b271cd7b7b891185e180376 Mon Sep 17 00:00:00 2001 From: Mara Date: Thu, 4 Jan 2024 11:27:55 +0100 Subject: [PATCH 10/11] respect guildeline --- src/index.js | 350 +----------------------------------------------- src/vec_lite.js | 3 +- 2 files changed, 3 insertions(+), 350 deletions(-) diff --git a/src/index.js b/src/index.js index c2e69efc..1b85db28 100644 --- a/src/index.js +++ b/src/index.js @@ -1,351 +1,5 @@ const Obsidian = require("obsidian"); -const VecLite = class { - constructor(config) { - this.config = { - file_name: "embeddings-3.json", - folder_path: ".vec_lite", - exists_adapter: null, - mkdir_adapter: null, - read_adapter: null, - rename_adapter: null, - stat_adapter: null, - write_adapter: null, - ...config, - }; - this.file_name = this.config.file_name; - this.folder_path = config.folder_path; - this.file_path = this.folder_path + "/" + this.file_name; - this.embeddings = false; - } - async file_exists(path) { - if (this.config.exists_adapter) { - return await this.config.exists_adapter(path); - } else { - throw new Error("exists_adapter not set"); - } - } - async mkdir(path) { - if (this.config.mkdir_adapter) { - return await this.config.mkdir_adapter(path); - } else { - throw new Error("mkdir_adapter not set"); - } - } - async read_file(path) { - if (this.config.read_adapter) { - return await this.config.read_adapter(path); - } else { - throw new Error("read_adapter not set"); - } - } - async rename(old_path, new_path) { - if (this.config.rename_adapter) { - return await this.config.rename_adapter(old_path, new_path); - } else { - throw new Error("rename_adapter not set"); - } - } - async stat(path) { - if (this.config.stat_adapter) { - return await this.config.stat_adapter(path); - } else { - throw new Error("stat_adapter not set"); - } - } - async write_file(path, data) { - if (this.config.write_adapter) { - return await this.config.write_adapter(path, data); - } else { - throw new Error("write_adapter not set"); - } - } - async load(retries = 0) { - try { - const embeddings_file = await this.read_file(this.file_path); - this.embeddings = JSON.parse(embeddings_file); - console.log("loaded embeddings file: " + this.file_path); - return true; - } catch (error) { - if (retries < 3) { - console.log("retrying load()"); - await new Promise((r) => setTimeout(r, 1e3 + 1e3 * retries)); - return await this.load(retries + 1); - } - console.log( - "failed to load embeddings file, prompt user to initiate bulk embed" - ); - return false; - } - } - async init_embeddings_file() { - if (!(await this.file_exists(this.folder_path))) { - await this.mkdir(this.folder_path); - console.log("created folder: " + this.folder_path); - } else { - console.log("folder already exists: " + this.folder_path); - } - if (!(await this.file_exists(this.file_path))) { - await this.write_file(this.file_path, "{}"); - console.log("created embeddings file: " + this.file_path); - } else { - console.log("embeddings file already exists: " + this.file_path); - } - } - async save() { - const embeddings = JSON.stringify(this.embeddings); - const embeddings_file_exists = await this.file_exists(this.file_path); - if (embeddings_file_exists) { - const new_file_size = embeddings.length; - const existing_file_size = await this.stat(this.file_path).then( - (stat) => stat.size - ); - if (new_file_size > existing_file_size * 0.5) { - await this.write_file(this.file_path, embeddings); - console.log("embeddings file size: " + new_file_size + " bytes"); - } else { - const warning_message = [ - "Warning: New embeddings file size is significantly smaller than existing embeddings file size.", - "Aborting to prevent possible loss of embeddings data.", - "New file size: " + new_file_size + " bytes.", - "Existing file size: " + existing_file_size + " bytes.", - "Restarting Obsidian may fix this.", - ]; - console.log(warning_message.join(" ")); - await this.write_file( - this.folder_path + "/unsaved-embeddings.json", - embeddings - ); - throw new Error( - "Error: New embeddings file size is significantly smaller than existing embeddings file size. Aborting to prevent possible loss of embeddings data." - ); - } - } else { - await this.init_embeddings_file(); - return await this.save(); - } - return true; - } - cos_sim(vector1, vector2) { - let dotProduct = 0; - let normA = 0; - let normB = 0; - for (let i = 0; i < vector1.length; i++) { - dotProduct += vector1[i] * vector2[i]; - normA += vector1[i] * vector1[i]; - normB += vector2[i] * vector2[i]; - } - if (normA === 0 || normB === 0) { - return 0; - } else { - return dotProduct / (Math.sqrt(normA) * Math.sqrt(normB)); - } - } - nearest(to_vec, filter = {}) { - filter = { - results_count: 30, - ...filter, - }; - let nearest = []; - const from_keys = Object.keys(this.embeddings); - for (let i = 0; i < from_keys.length; i++) { - if (filter.skip_sections) { - const from_path = this.embeddings[from_keys[i]].meta.path; - if (from_path.indexOf("#") > -1) continue; - } - if (filter.skip_key) { - if (filter.skip_key === from_keys[i]) continue; - if (filter.skip_key === this.embeddings[from_keys[i]].meta.parent) - continue; - } - if (filter.path_begins_with) { - if ( - typeof filter.path_begins_with === "string" && - !this.embeddings[from_keys[i]].meta.path.startsWith( - filter.path_begins_with - ) - ) - continue; - if ( - Array.isArray(filter.path_begins_with) && - !filter.path_begins_with.some((path) => - this.embeddings[from_keys[i]].meta.path.startsWith(path) - ) - ) - continue; - } - nearest.push({ - link: this.embeddings[from_keys[i]].meta.path, - similarity: this.cos_sim(to_vec, this.embeddings[from_keys[i]].vec), - size: this.embeddings[from_keys[i]].meta.size, - }); - } - nearest.sort(function (a, b) { - return b.similarity - a.similarity; - }); - nearest = nearest.slice(0, filter.results_count); - return nearest; - } - find_nearest_embeddings(to_vec, filter = {}) { - const default_filter = { - max: this.max_sources, - }; - filter = { ...default_filter, ...filter }; - if (Array.isArray(to_vec) && to_vec.length !== this.vec_len) { - this.nearest = {}; - for (let i = 0; i < to_vec.length; i++) { - this.find_nearest_embeddings(to_vec[i], { - max: Math.floor(filter.max / to_vec.length), - }); - } - } else { - const from_keys = Object.keys(this.embeddings); - for (let i = 0; i < from_keys.length; i++) { - if (this.validate_type(this.embeddings[from_keys[i]])) continue; - const sim = this.computeCosineSimilarity( - to_vec, - this.embeddings[from_keys[i]].vec - ); - if (this.nearest[from_keys[i]]) { - this.nearest[from_keys[i]] += sim; - } else { - this.nearest[from_keys[i]] = sim; - } - } - } - let nearest = Object.keys(this.nearest).map((key) => { - return { - key, - similarity: this.nearest[key], - }; - }); - nearest = this.sort_by_similarity(nearest); - nearest = nearest.slice(0, filter.max); - nearest = nearest.map((item) => { - return { - link: this.embeddings[item.key].meta.path, - similarity: item.similarity, - len: - this.embeddings[item.key].meta.len || - this.embeddings[item.key].meta.size, - }; - }); - return nearest; - } - sort_by_similarity(nearest) { - return nearest.sort(function (a, b) { - const a_score = a.similarity; - const b_score = b.similarity; - if (a_score > b_score) return -1; - if (a_score < b_score) return 1; - return 0; - }); - } - // check if key from embeddings exists in files - clean_up_embeddings(files) { - console.log("cleaning up embeddings"); - const keys = Object.keys(this.embeddings); - let deleted_embeddings = 0; - for (const key of keys) { - const path = this.embeddings[key].meta.path; - if (!files.find((file) => path.startsWith(file.path))) { - delete this.embeddings[key]; - deleted_embeddings++; - continue; - } - if (path.indexOf("#") > -1) { - const parent_key = this.embeddings[key].meta.parent; - if (!this.embeddings[parent_key]) { - delete this.embeddings[key]; - deleted_embeddings++; - continue; - } - if (!this.embeddings[parent_key].meta) { - delete this.embeddings[key]; - deleted_embeddings++; - continue; - } - if ( - this.embeddings[parent_key].meta.children && - this.embeddings[parent_key].meta.children.indexOf(key) < 0 - ) { - delete this.embeddings[key]; - deleted_embeddings++; - continue; - } - } - } - return { deleted_embeddings, total_embeddings: keys.length }; - } - get(key) { - return this.embeddings[key] || null; - } - get_meta(key) { - const embedding = this.get(key); - if (embedding && embedding.meta) { - return embedding.meta; - } - return null; - } - get_mtime(key) { - const meta = this.get_meta(key); - if (meta && meta.mtime) { - return meta.mtime; - } - return null; - } - get_hash(key) { - const meta = this.get_meta(key); - if (meta && meta.hash) { - return meta.hash; - } - return null; - } - get_size(key) { - const meta = this.get_meta(key); - if (meta && meta.size) { - return meta.size; - } - return null; - } - get_children(key) { - const meta = this.get_meta(key); - if (meta && meta.children) { - return meta.children; - } - return null; - } - get_vec(key) { - const embedding = this.get(key); - if (embedding && embedding.vec) { - return embedding.vec; - } - return null; - } - save_embedding(key, vec, meta) { - this.embeddings[key] = { - vec, - meta, - }; - } - mtime_is_current(key, source_mtime) { - const mtime = this.get_mtime(key); - if (mtime && mtime >= source_mtime) { - return true; - } - return false; - } - async force_refresh() { - this.embeddings = null; - this.embeddings = {}; - let current_datetime = Math.floor(Date.now() / 1e3); - await this.rename( - this.file_path, - this.folder_path + "/embeddings-" + current_datetime + ".json" - ); - await this.init_embeddings_file(); - } -}; - +const VecLite = require("./vec_lite"); const DEFAULT_SETTINGS = { @@ -451,8 +105,6 @@ class SmartConnectionsPlugin extends Obsidian.Plugin { onunload() { this.output_render_log(); console.log("unloading plugin"); - this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_VIEW_TYPE); - this.app.workspace.detachLeavesOfType(SMART_CONNECTIONS_CHAT_VIEW_TYPE); } async initialize() { console.log("Loading Smart Connections plugin"); diff --git a/src/vec_lite.js b/src/vec_lite.js index cc301f79..437b6259 100644 --- a/src/vec_lite.js +++ b/src/vec_lite.js @@ -1,4 +1,4 @@ -export default class { +const Veclite = class { constructor(config) { this.config = { file_name: "embeddings-3.json", @@ -345,3 +345,4 @@ export default class { } }; +export default Veclite; \ No newline at end of file From 588e0272256c0ec9fad3f3c394a75d446d1e3bc3 Mon Sep 17 00:00:00 2001 From: Mara Date: Thu, 4 Jan 2024 11:28:51 +0100 Subject: [PATCH 11/11] chore: gitignore --- .gitignore | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/.gitignore b/.gitignore index c395c041..90e8a579 100644 --- a/.gitignore +++ b/.gitignore @@ -8,4 +8,5 @@ package-lock.json # symlinks for context obsidian-plugins notes -vec_lite \ No newline at end of file +vec_lite +dist/* \ No newline at end of file