/* THIS IS A GENERATED/BUNDLED FILE BY ESBUILD if you want to view the source, please visit the github repository of this plugin */ var __create = Object.create; var __defProp = Object.defineProperty; var __getOwnPropDesc = Object.getOwnPropertyDescriptor; var __getOwnPropNames = Object.getOwnPropertyNames; var __getProtoOf = Object.getPrototypeOf; var __hasOwnProp = Object.prototype.hasOwnProperty; var __esm = (fn, res) => function __init() { return fn && (res = (0, fn[__getOwnPropNames(fn)[0]])(fn = 0)), res; }; var __commonJS = (cb, mod) => function __require() { return mod || (0, cb[__getOwnPropNames(cb)[0]])((mod = { exports: {} }).exports, mod), mod.exports; }; var __export = (target, all) => { for (var name in all) __defProp(target, name, { get: all[name], enumerable: true }); }; var __copyProps = (to, from, except, desc) => { if (from && typeof from === "object" || typeof from === "function") { for (let key of __getOwnPropNames(from)) if (!__hasOwnProp.call(to, key) && key !== except) __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable }); } return to; }; var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps( // If the importer is in node compatibility mode or this is not an ESM // file that has been converted to a CommonJS file using a Babel- // compatible transform (i.e. "__esModule" has not been set), then set // "default" to the CommonJS "module.exports" for node compatibility. isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target, mod )); var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod); // node_modules/argon2-browser/dist/argon2.js var require_argon2 = __commonJS({ "node_modules/argon2-browser/dist/argon2.js"(exports, module2) { var Module2 = typeof self !== "undefined" && typeof self.Module !== "undefined" ? self.Module : {}; var jsModule = Module2; var moduleOverrides = {}; var key; for (key in Module2) { if (Module2.hasOwnProperty(key)) { moduleOverrides[key] = Module2[key]; } } var arguments_ = []; var thisProgram = "./this.program"; var quit_ = function(status, toThrow) { throw toThrow; }; var ENVIRONMENT_IS_WEB = false; var ENVIRONMENT_IS_WORKER = false; var ENVIRONMENT_IS_NODE = false; var ENVIRONMENT_IS_SHELL = false; ENVIRONMENT_IS_WEB = typeof window === "object"; ENVIRONMENT_IS_WORKER = typeof importScripts === "function"; ENVIRONMENT_IS_NODE = typeof process === "object" && typeof process.versions === "object" && typeof process.versions.node === "string"; ENVIRONMENT_IS_SHELL = !ENVIRONMENT_IS_WEB && !ENVIRONMENT_IS_NODE && !ENVIRONMENT_IS_WORKER; var scriptDirectory = ""; function locateFile(path) { if (Module2["locateFile"]) { return Module2["locateFile"](path, scriptDirectory); } return scriptDirectory + path; } var read_; var readAsync; var readBinary; var setWindowTitle; var nodeFS; var nodePath; if (ENVIRONMENT_IS_NODE) { if (ENVIRONMENT_IS_WORKER) { scriptDirectory = require("path").dirname(scriptDirectory) + "/"; } else { scriptDirectory = __dirname + "/"; } read_ = function shell_read(filename, binary) { if (!nodeFS) nodeFS = require("fs"); if (!nodePath) nodePath = require("path"); filename = nodePath["normalize"](filename); return nodeFS["readFileSync"](filename, binary ? null : "utf8"); }; readBinary = function readBinary2(filename) { var ret = read_(filename, true); if (!ret.buffer) { ret = new Uint8Array(ret); } assert(ret.buffer); return ret; }; if (process["argv"].length > 1) { thisProgram = process["argv"][1].replace(/\\/g, "/"); } arguments_ = process["argv"].slice(2); if (typeof module2 !== "undefined") { module2["exports"] = Module2; } process["on"]("uncaughtException", function(ex) { if (!(ex instanceof ExitStatus)) { throw ex; } }); process["on"]("unhandledRejection", abort); quit_ = function(status) { process["exit"](status); }; Module2["inspect"] = function() { return "[Emscripten Module object]"; }; } else if (ENVIRONMENT_IS_SHELL) { if (typeof read != "undefined") { read_ = function shell_read(f) { return read(f); }; } readBinary = function readBinary2(f) { var data; if (typeof readbuffer === "function") { return new Uint8Array(readbuffer(f)); } data = read(f, "binary"); assert(typeof data === "object"); return data; }; if (typeof scriptArgs != "undefined") { arguments_ = scriptArgs; } else if (typeof arguments != "undefined") { arguments_ = arguments; } if (typeof quit === "function") { quit_ = function(status) { quit(status); }; } if (typeof print !== "undefined") { if (typeof console === "undefined") console = {}; console.log = print; console.warn = console.error = typeof printErr !== "undefined" ? printErr : print; } } else if (ENVIRONMENT_IS_WEB || ENVIRONMENT_IS_WORKER) { if (ENVIRONMENT_IS_WORKER) { scriptDirectory = self.location.href; } else if (typeof document !== "undefined" && document.currentScript) { scriptDirectory = document.currentScript.src; } if (scriptDirectory.indexOf("blob:") !== 0) { scriptDirectory = scriptDirectory.substr(0, scriptDirectory.lastIndexOf("/") + 1); } else { scriptDirectory = ""; } { read_ = function(url) { var xhr = new XMLHttpRequest(); xhr.open("GET", url, false); xhr.send(null); return xhr.responseText; }; if (ENVIRONMENT_IS_WORKER) { readBinary = function(url) { var xhr = new XMLHttpRequest(); xhr.open("GET", url, false); xhr.responseType = "arraybuffer"; xhr.send(null); return new Uint8Array(xhr.response); }; } readAsync = function(url, onload, onerror) { var xhr = new XMLHttpRequest(); xhr.open("GET", url, true); xhr.responseType = "arraybuffer"; xhr.onload = function() { if (xhr.status == 200 || xhr.status == 0 && xhr.response) { onload(xhr.response); return; } onerror(); }; xhr.onerror = onerror; xhr.send(null); }; } setWindowTitle = function(title) { document.title = title; }; } else { } var out = Module2["print"] || console.log.bind(console); var err = Module2["printErr"] || console.warn.bind(console); for (key in moduleOverrides) { if (moduleOverrides.hasOwnProperty(key)) { Module2[key] = moduleOverrides[key]; } } moduleOverrides = null; if (Module2["arguments"]) arguments_ = Module2["arguments"]; if (Module2["thisProgram"]) thisProgram = Module2["thisProgram"]; if (Module2["quit"]) quit_ = Module2["quit"]; var wasmBinary; if (Module2["wasmBinary"]) wasmBinary = Module2["wasmBinary"]; var noExitRuntime = Module2["noExitRuntime"] || true; if (typeof WebAssembly !== "object") { abort("no native wasm support detected"); } var wasmMemory; var ABORT = false; var EXITSTATUS; function assert(condition, text) { if (!condition) { abort("Assertion failed: " + text); } } var ALLOC_NORMAL = 0; var ALLOC_STACK = 1; function allocate(slab, allocator) { var ret; if (allocator == ALLOC_STACK) { ret = stackAlloc(slab.length); } else { ret = _malloc(slab.length); } if (slab.subarray || slab.slice) { HEAPU8.set(slab, ret); } else { HEAPU8.set(new Uint8Array(slab), ret); } return ret; } var UTF8Decoder = typeof TextDecoder !== "undefined" ? new TextDecoder("utf8") : void 0; function UTF8ArrayToString(heap, idx, maxBytesToRead) { var endIdx = idx + maxBytesToRead; var endPtr = idx; while (heap[endPtr] && !(endPtr >= endIdx)) ++endPtr; if (endPtr - idx > 16 && heap.subarray && UTF8Decoder) { return UTF8Decoder.decode(heap.subarray(idx, endPtr)); } else { var str = ""; while (idx < endPtr) { var u0 = heap[idx++]; if (!(u0 & 128)) { str += String.fromCharCode(u0); continue; } var u1 = heap[idx++] & 63; if ((u0 & 224) == 192) { str += String.fromCharCode((u0 & 31) << 6 | u1); continue; } var u2 = heap[idx++] & 63; if ((u0 & 240) == 224) { u0 = (u0 & 15) << 12 | u1 << 6 | u2; } else { u0 = (u0 & 7) << 18 | u1 << 12 | u2 << 6 | heap[idx++] & 63; } if (u0 < 65536) { str += String.fromCharCode(u0); } else { var ch = u0 - 65536; str += String.fromCharCode(55296 | ch >> 10, 56320 | ch & 1023); } } } return str; } function UTF8ToString(ptr, maxBytesToRead) { return ptr ? UTF8ArrayToString(HEAPU8, ptr, maxBytesToRead) : ""; } function alignUp(x, multiple) { if (x % multiple > 0) { x += multiple - x % multiple; } return x; } var buffer; var HEAP8; var HEAPU8; var HEAP16; var HEAPU16; var HEAP32; var HEAPU32; var HEAPF32; var HEAPF64; function updateGlobalBufferAndViews(buf) { buffer = buf; Module2["HEAP8"] = HEAP8 = new Int8Array(buf); Module2["HEAP16"] = HEAP16 = new Int16Array(buf); Module2["HEAP32"] = HEAP32 = new Int32Array(buf); Module2["HEAPU8"] = HEAPU8 = new Uint8Array(buf); Module2["HEAPU16"] = HEAPU16 = new Uint16Array(buf); Module2["HEAPU32"] = HEAPU32 = new Uint32Array(buf); Module2["HEAPF32"] = HEAPF32 = new Float32Array(buf); Module2["HEAPF64"] = HEAPF64 = new Float64Array(buf); } var INITIAL_MEMORY = Module2["INITIAL_MEMORY"] || 16777216; var wasmTable; var __ATPRERUN__ = []; var __ATINIT__ = []; var __ATPOSTRUN__ = []; var runtimeInitialized = false; function preRun() { if (Module2["preRun"]) { if (typeof Module2["preRun"] == "function") Module2["preRun"] = [Module2["preRun"]]; while (Module2["preRun"].length) { addOnPreRun(Module2["preRun"].shift()); } } callRuntimeCallbacks(__ATPRERUN__); } function initRuntime() { runtimeInitialized = true; callRuntimeCallbacks(__ATINIT__); } function postRun() { if (Module2["postRun"]) { if (typeof Module2["postRun"] == "function") Module2["postRun"] = [Module2["postRun"]]; while (Module2["postRun"].length) { addOnPostRun(Module2["postRun"].shift()); } } callRuntimeCallbacks(__ATPOSTRUN__); } function addOnPreRun(cb) { __ATPRERUN__.unshift(cb); } function addOnInit(cb) { __ATINIT__.unshift(cb); } function addOnPostRun(cb) { __ATPOSTRUN__.unshift(cb); } var runDependencies = 0; var runDependencyWatcher = null; var dependenciesFulfilled = null; function addRunDependency(id) { runDependencies++; if (Module2["monitorRunDependencies"]) { Module2["monitorRunDependencies"](runDependencies); } } function removeRunDependency(id) { runDependencies--; if (Module2["monitorRunDependencies"]) { Module2["monitorRunDependencies"](runDependencies); } if (runDependencies == 0) { if (runDependencyWatcher !== null) { clearInterval(runDependencyWatcher); runDependencyWatcher = null; } if (dependenciesFulfilled) { var callback = dependenciesFulfilled; dependenciesFulfilled = null; callback(); } } } Module2["preloadedImages"] = {}; Module2["preloadedAudios"] = {}; function abort(what) { if (Module2["onAbort"]) { Module2["onAbort"](what); } what += ""; err(what); ABORT = true; EXITSTATUS = 1; what = "abort(" + what + "). Build with -s ASSERTIONS=1 for more info."; var e = new WebAssembly.RuntimeError(what); throw e; } var dataURIPrefix = "data:application/octet-stream;base64,"; function isDataURI(filename) { return filename.startsWith(dataURIPrefix); } function isFileURI(filename) { return filename.startsWith("file://"); } var wasmBinaryFile = "argon2.wasm"; if (!isDataURI(wasmBinaryFile)) { wasmBinaryFile = locateFile(wasmBinaryFile); } function getBinary(file) { try { if (file == wasmBinaryFile && wasmBinary) { return new Uint8Array(wasmBinary); } if (readBinary) { return readBinary(file); } else { throw "both async and sync fetching of the wasm failed"; } } catch (err2) { abort(err2); } } function getBinaryPromise() { if (!wasmBinary && (ENVIRONMENT_IS_WEB || ENVIRONMENT_IS_WORKER)) { if (typeof fetch === "function" && !isFileURI(wasmBinaryFile)) { return fetch(wasmBinaryFile, { credentials: "same-origin" }).then(function(response) { if (!response["ok"]) { throw "failed to load wasm binary file at '" + wasmBinaryFile + "'"; } return response["arrayBuffer"](); }).catch(function() { return getBinary(wasmBinaryFile); }); } else { if (readAsync) { return new Promise(function(resolve, reject) { readAsync(wasmBinaryFile, function(response) { resolve(new Uint8Array(response)); }, reject); }); } } } return Promise.resolve().then(function() { return getBinary(wasmBinaryFile); }); } function createWasm() { var info = { "a": asmLibraryArg }; function receiveInstance(instance, module3) { var exports3 = instance.exports; Module2["asm"] = exports3; wasmMemory = Module2["asm"]["c"]; updateGlobalBufferAndViews(wasmMemory.buffer); wasmTable = Module2["asm"]["k"]; addOnInit(Module2["asm"]["d"]); removeRunDependency("wasm-instantiate"); } addRunDependency("wasm-instantiate"); function receiveInstantiationResult(result) { receiveInstance(result["instance"]); } function instantiateArrayBuffer(receiver) { return getBinaryPromise().then(function(binary) { var result = WebAssembly.instantiate(binary, info); return result; }).then(receiver, function(reason) { err("failed to asynchronously prepare wasm: " + reason); abort(reason); }); } function instantiateAsync() { if (!wasmBinary && typeof WebAssembly.instantiateStreaming === "function" && !isDataURI(wasmBinaryFile) && !isFileURI(wasmBinaryFile) && typeof fetch === "function") { return fetch(wasmBinaryFile, { credentials: "same-origin" }).then(function(response) { var result = WebAssembly.instantiateStreaming(response, info); return result.then(receiveInstantiationResult, function(reason) { err("wasm streaming compile failed: " + reason); err("falling back to ArrayBuffer instantiation"); return instantiateArrayBuffer(receiveInstantiationResult); }); }); } else { return instantiateArrayBuffer(receiveInstantiationResult); } } if (Module2["instantiateWasm"]) { try { var exports2 = Module2["instantiateWasm"](info, receiveInstance); return exports2; } catch (e) { err("Module.instantiateWasm callback failed with error: " + e); return false; } } instantiateAsync(); return {}; } function callRuntimeCallbacks(callbacks) { while (callbacks.length > 0) { var callback = callbacks.shift(); if (typeof callback == "function") { callback(Module2); continue; } var func = callback.func; if (typeof func === "number") { if (callback.arg === void 0) { wasmTable.get(func)(); } else { wasmTable.get(func)(callback.arg); } } else { func(callback.arg === void 0 ? null : callback.arg); } } } function _emscripten_memcpy_big(dest, src, num) { HEAPU8.copyWithin(dest, src, src + num); } function emscripten_realloc_buffer(size) { try { wasmMemory.grow(size - buffer.byteLength + 65535 >>> 16); updateGlobalBufferAndViews(wasmMemory.buffer); return 1; } catch (e) { } } function _emscripten_resize_heap(requestedSize) { var oldSize = HEAPU8.length; requestedSize = requestedSize >>> 0; var maxHeapSize = 2147418112; if (requestedSize > maxHeapSize) { return false; } for (var cutDown = 1; cutDown <= 4; cutDown *= 2) { var overGrownHeapSize = oldSize * (1 + 0.2 / cutDown); overGrownHeapSize = Math.min(overGrownHeapSize, requestedSize + 100663296); var newSize = Math.min(maxHeapSize, alignUp(Math.max(requestedSize, overGrownHeapSize), 65536)); var replacement = emscripten_realloc_buffer(newSize); if (replacement) { return true; } } return false; } var asmLibraryArg = { "a": _emscripten_memcpy_big, "b": _emscripten_resize_heap }; var asm = createWasm(); var ___wasm_call_ctors = Module2["___wasm_call_ctors"] = function() { return (___wasm_call_ctors = Module2["___wasm_call_ctors"] = Module2["asm"]["d"]).apply(null, arguments); }; var _argon2_hash = Module2["_argon2_hash"] = function() { return (_argon2_hash = Module2["_argon2_hash"] = Module2["asm"]["e"]).apply(null, arguments); }; var _malloc = Module2["_malloc"] = function() { return (_malloc = Module2["_malloc"] = Module2["asm"]["f"]).apply(null, arguments); }; var _free = Module2["_free"] = function() { return (_free = Module2["_free"] = Module2["asm"]["g"]).apply(null, arguments); }; var _argon2_verify = Module2["_argon2_verify"] = function() { return (_argon2_verify = Module2["_argon2_verify"] = Module2["asm"]["h"]).apply(null, arguments); }; var _argon2_error_message = Module2["_argon2_error_message"] = function() { return (_argon2_error_message = Module2["_argon2_error_message"] = Module2["asm"]["i"]).apply(null, arguments); }; var _argon2_encodedlen = Module2["_argon2_encodedlen"] = function() { return (_argon2_encodedlen = Module2["_argon2_encodedlen"] = Module2["asm"]["j"]).apply(null, arguments); }; var _argon2_hash_ext = Module2["_argon2_hash_ext"] = function() { return (_argon2_hash_ext = Module2["_argon2_hash_ext"] = Module2["asm"]["l"]).apply(null, arguments); }; var _argon2_verify_ext = Module2["_argon2_verify_ext"] = function() { return (_argon2_verify_ext = Module2["_argon2_verify_ext"] = Module2["asm"]["m"]).apply(null, arguments); }; var stackAlloc = Module2["stackAlloc"] = function() { return (stackAlloc = Module2["stackAlloc"] = Module2["asm"]["n"]).apply(null, arguments); }; Module2["allocate"] = allocate; Module2["UTF8ToString"] = UTF8ToString; Module2["ALLOC_NORMAL"] = ALLOC_NORMAL; var calledRun; function ExitStatus(status) { this.name = "ExitStatus"; this.message = "Program terminated with exit(" + status + ")"; this.status = status; } dependenciesFulfilled = function runCaller() { if (!calledRun) run(); if (!calledRun) dependenciesFulfilled = runCaller; }; function run(args) { args = args || arguments_; if (runDependencies > 0) { return; } preRun(); if (runDependencies > 0) { return; } function doRun() { if (calledRun) return; calledRun = true; Module2["calledRun"] = true; if (ABORT) return; initRuntime(); if (Module2["onRuntimeInitialized"]) Module2["onRuntimeInitialized"](); postRun(); } if (Module2["setStatus"]) { Module2["setStatus"]("Running..."); setTimeout(function() { setTimeout(function() { Module2["setStatus"](""); }, 1); doRun(); }, 1); } else { doRun(); } } Module2["run"] = run; if (Module2["preInit"]) { if (typeof Module2["preInit"] == "function") Module2["preInit"] = [Module2["preInit"]]; while (Module2["preInit"].length > 0) { Module2["preInit"].pop()(); } } run(); if (typeof module2 !== "undefined") module2.exports = Module2; Module2.unloadRuntime = function() { if (typeof self !== "undefined") { delete self.Module; } Module2 = jsModule = wasmMemory = wasmTable = asm = buffer = HEAP8 = HEAPU8 = HEAP16 = HEAPU16 = HEAP32 = HEAPU32 = HEAPF32 = HEAPF64 = void 0; if (typeof module2 !== "undefined") { delete module2.exports; } }; } }); // node_modules/argon2-browser/dist/argon2.wasm var require_argon22 = __commonJS({ "node_modules/argon2-browser/dist/argon2.wasm"(exports, module2) { module2.exports = "./assets/argon2.wasm"; } }); // node_modules/argon2-browser/lib/argon2.js var require_argon23 = __commonJS({ "node_modules/argon2-browser/lib/argon2.js"(exports, module2) { (function(root, factory) { if (typeof define === "function" && define.amd) { define([], factory); } else if (typeof module2 === "object" && module2.exports) { module2.exports = factory(); } else { root.argon2 = factory(); } })(typeof self !== "undefined" ? self : exports, function() { const global = typeof self !== "undefined" ? self : this; const ArgonType = { Argon2d: 0, Argon2i: 1, Argon2id: 2 }; function loadModule(mem) { if (loadModule._promise) { return loadModule._promise; } if (loadModule._module) { return Promise.resolve(loadModule._module); } let promise; if (global.process && global.process.versions && global.process.versions.node) { promise = loadWasmModule().then( (Module2) => new Promise((resolve) => { Module2.postRun = () => resolve(Module2); }) ); } else { promise = loadWasmBinary().then((wasmBinary) => { const wasmMemory = mem ? createWasmMemory(mem) : void 0; return initWasm(wasmBinary, wasmMemory); }); } loadModule._promise = promise; return promise.then((Module2) => { loadModule._module = Module2; delete loadModule._promise; return Module2; }); } function initWasm(wasmBinary, wasmMemory) { return new Promise((resolve) => { global.Module = { wasmBinary, wasmMemory, postRun() { resolve(Module); } }; return loadWasmModule(); }); } function loadWasmModule() { if (global.loadArgon2WasmModule) { return global.loadArgon2WasmModule(); } if (typeof require === "function") { return Promise.resolve(require_argon2()); } return Promise.resolve().then(() => __toESM(require_argon2())); } function loadWasmBinary() { if (global.loadArgon2WasmBinary) { return global.loadArgon2WasmBinary(); } if (typeof require === "function") { return Promise.resolve(require_argon22()).then( (wasmModule) => { return decodeWasmBinary(wasmModule); } ); } const wasmPath = global.argon2WasmPath || "node_modules/argon2-browser/dist/argon2.wasm"; return fetch(wasmPath).then((response) => response.arrayBuffer()).then((ab) => new Uint8Array(ab)); } function decodeWasmBinary(base64) { const text = atob(base64); const binary = new Uint8Array(new ArrayBuffer(text.length)); for (let i = 0; i < text.length; i++) { binary[i] = text.charCodeAt(i); } return binary; } function createWasmMemory(mem) { const KB = 1024; const MB = 1024 * KB; const GB = 1024 * MB; const WASM_PAGE_SIZE = 64 * KB; const totalMemory = (2 * GB - 64 * KB) / WASM_PAGE_SIZE; const initialMemory = Math.min( Math.max(Math.ceil(mem * KB / WASM_PAGE_SIZE), 256) + 256, totalMemory ); return new WebAssembly.Memory({ initial: initialMemory, maximum: totalMemory }); } function allocateArray(Module2, arr) { return Module2.allocate(arr, "i8", Module2.ALLOC_NORMAL); } function allocateArrayStr(Module2, arr) { const nullTerminatedArray = new Uint8Array([...arr, 0]); return allocateArray(Module2, nullTerminatedArray); } function encodeUtf8(str) { if (typeof str !== "string") { return str; } if (typeof TextEncoder === "function") { return new TextEncoder().encode(str); } else if (typeof Buffer === "function") { return Buffer.from(str); } else { throw new Error("Don't know how to encode UTF8"); } } function argon2Hash(params) { const mCost = params.mem || 1024; return loadModule(mCost).then((Module2) => { const tCost = params.time || 1; const parallelism = params.parallelism || 1; const pwdEncoded = encodeUtf8(params.pass); const pwd = allocateArrayStr(Module2, pwdEncoded); const pwdlen = pwdEncoded.length; const saltEncoded = encodeUtf8(params.salt); const salt = allocateArrayStr(Module2, saltEncoded); const saltlen = saltEncoded.length; const argon2Type = params.type || ArgonType.Argon2d; const hash = Module2.allocate( new Array(params.hashLen || 24), "i8", Module2.ALLOC_NORMAL ); const secret = params.secret ? allocateArray(Module2, params.secret) : 0; const secretlen = params.secret ? params.secret.byteLength : 0; const ad = params.ad ? allocateArray(Module2, params.ad) : 0; const adlen = params.ad ? params.ad.byteLength : 0; const hashlen = params.hashLen || 24; const encodedlen = Module2._argon2_encodedlen( tCost, mCost, parallelism, saltlen, hashlen, argon2Type ); const encoded = Module2.allocate( new Array(encodedlen + 1), "i8", Module2.ALLOC_NORMAL ); const version = 19; let err; let res; try { res = Module2._argon2_hash_ext( tCost, mCost, parallelism, pwd, pwdlen, salt, saltlen, hash, hashlen, encoded, encodedlen, argon2Type, secret, secretlen, ad, adlen, version ); } catch (e) { err = e; } let result; if (res === 0 && !err) { let hashStr = ""; const hashArr = new Uint8Array(hashlen); for (let i = 0; i < hashlen; i++) { const byte = Module2.HEAP8[hash + i]; hashArr[i] = byte; hashStr += ("0" + (255 & byte).toString(16)).slice(-2); } const encodedStr = Module2.UTF8ToString(encoded); result = { hash: hashArr, hashHex: hashStr, encoded: encodedStr }; } else { try { if (!err) { err = Module2.UTF8ToString( Module2._argon2_error_message(res) ); } } catch (e) { } result = { message: err, code: res }; } try { Module2._free(pwd); Module2._free(salt); Module2._free(hash); Module2._free(encoded); if (ad) { Module2._free(ad); } if (secret) { Module2._free(secret); } } catch (e) { } if (err) { throw result; } else { return result; } }); } function argon2Verify(params) { return loadModule().then((Module2) => { const pwdEncoded = encodeUtf8(params.pass); const pwd = allocateArrayStr(Module2, pwdEncoded); const pwdlen = pwdEncoded.length; const secret = params.secret ? allocateArray(Module2, params.secret) : 0; const secretlen = params.secret ? params.secret.byteLength : 0; const ad = params.ad ? allocateArray(Module2, params.ad) : 0; const adlen = params.ad ? params.ad.byteLength : 0; const encEncoded = encodeUtf8(params.encoded); const enc = allocateArrayStr(Module2, encEncoded); let argon2Type = params.type; if (argon2Type === void 0) { let typeStr = params.encoded.split("$")[1]; if (typeStr) { typeStr = typeStr.replace("a", "A"); argon2Type = ArgonType[typeStr] || ArgonType.Argon2d; } } let err; let res; try { res = Module2._argon2_verify_ext( enc, pwd, pwdlen, secret, secretlen, ad, adlen, argon2Type ); } catch (e) { err = e; } let result; if (res || err) { try { if (!err) { err = Module2.UTF8ToString( Module2._argon2_error_message(res) ); } } catch (e) { } result = { message: err, code: res }; } try { Module2._free(pwd); Module2._free(enc); } catch (e) { } if (err) { throw result; } else { return result; } }); } function unloadRuntime() { if (loadModule._module) { loadModule._module.unloadRuntime(); delete loadModule._promise; delete loadModule._module; } } return { ArgonType, hash: argon2Hash, verify: argon2Verify, unloadRuntime }; }); } }); // src/encryption/aesWithPassword.ts var aesWithPassword_exports = {}; __export(aesWithPassword_exports, { decryptBytesWithPassword: () => decryptBytesWithPassword, decryptWithPassword: () => decryptWithPassword, encryptBytesWithPassword: () => encryptBytesWithPassword, encryptWithPassword: () => encryptWithPassword }); async function encryptWithPassword(password, text, keyCache, kdfOpts) { const encoder = new TextEncoder(); const salt = crypto.getRandomValues(new Uint8Array(16)); const iv = crypto.getRandomValues(new Uint8Array(12)); const key = keyCache ? await getOrDeriveKey(password, salt, keyCache, kdfOpts) : await deriveKeyDirect(password, salt, kdfOpts); const encrypted = await crypto.subtle.encrypt( { name: "AES-GCM", iv }, key, encoder.encode(text) ); return { salt: bufferToBase64(salt), iv: bufferToBase64(iv), data: bufferToBase64(encrypted) }; } async function encryptBytesWithPassword(password, bytes, keyCache, kdfOpts) { const salt = crypto.getRandomValues(new Uint8Array(16)); const iv = crypto.getRandomValues(new Uint8Array(12)); const key = keyCache ? await getOrDeriveKey(password, salt, keyCache, kdfOpts) : await deriveKeyDirect(password, salt, kdfOpts); const data = bytes instanceof Uint8Array ? bytes : new Uint8Array(bytes); const encrypted = await crypto.subtle.encrypt( { name: "AES-GCM", iv }, key, data ); return { salt: bufferToBase64(salt), iv: bufferToBase64(iv), data: bufferToBase64(encrypted) }; } async function decryptWithPassword(password, saltB64, ivB64, dataB64, keyCache, kdfOpts) { const decoder = new TextDecoder(); const salt = base64ToBuffer(saltB64); const iv = base64ToBuffer(ivB64); const data = base64ToBuffer(dataB64); const key = keyCache ? await getOrDeriveKey(password, salt, keyCache, kdfOpts) : await deriveKeyDirect(password, salt, kdfOpts); try { const decrypted = await crypto.subtle.decrypt( { name: "AES-GCM", iv }, key, data ); return decoder.decode(decrypted); } catch (error) { console.error("AES decryption failed:", error); throw error; } } async function decryptBytesWithPassword(password, saltB64, ivB64, dataB64, keyCache, kdfOpts) { const salt = base64ToBuffer(saltB64); const iv = base64ToBuffer(ivB64); const data = base64ToBuffer(dataB64); const key = keyCache ? await getOrDeriveKey(password, salt, keyCache, kdfOpts) : await deriveKeyDirect(password, salt, kdfOpts); const decrypted = await crypto.subtle.decrypt( { name: "AES-GCM", iv }, key, data ); return new Uint8Array(decrypted); } async function getKeyMaterial(password) { const enc = new TextEncoder(); return crypto.subtle.importKey( "raw", enc.encode(password), { name: "PBKDF2" }, false, ["deriveBits", "deriveKey"] ); } async function deriveKeyPBKDF2(keyMaterial, salt, iterations) { return crypto.subtle.deriveKey( { name: "PBKDF2", salt, iterations, hash: "SHA-256" }, keyMaterial, { name: "AES-GCM", length: 256 }, false, ["encrypt", "decrypt"] ); } async function deriveKeyArgon2id(password, salt, kdfOpts) { var _a, _b, _c, _d, _e, _f, _g; const memKB = (_a = kdfOpts == null ? void 0 : kdfOpts.argon2MemoryKB) != null ? _a : 65536; const iters = (_b = kdfOpts == null ? void 0 : kdfOpts.argon2Iterations) != null ? _b : 3; const para = (_c = kdfOpts == null ? void 0 : kdfOpts.argon2Parallelism) != null ? _c : 1; const hashLen = (_d = kdfOpts == null ? void 0 : kdfOpts.argon2HashLen) != null ? _d : 32; try { const argon2 = await Promise.resolve().then(() => __toESM(require_argon23())); const res = await argon2.hash({ pass: password, salt, type: (_f = (_e = argon2.ArgonType) == null ? void 0 : _e.Argon2id) != null ? _f : 2, mem: memKB, time: iters, parallelism: para, hashLen, distPath: void 0 // let bundler handle assets }); const keyBytes = new Uint8Array(res.hash); return crypto.subtle.importKey( "raw", keyBytes, { name: "AES-GCM", length: 256 }, false, ["encrypt", "decrypt"] ); } catch (e) { const keyMaterial = await getKeyMaterial(password); const iterations = (_g = kdfOpts == null ? void 0 : kdfOpts.pbkdf2Iterations) != null ? _g : 6e5; return deriveKeyPBKDF2(keyMaterial, salt, iterations); } } function makeCachePasswordTag(password, kdfOpts) { var _a, _b, _c, _d, _e; if (!kdfOpts || kdfOpts.type === "PBKDF2") { const it = (_a = kdfOpts == null ? void 0 : kdfOpts.pbkdf2Iterations) != null ? _a : 6e5; return `pbkdf2:${it}:${password}`; } const memKB = (_b = kdfOpts.argon2MemoryKB) != null ? _b : 65536; const iters = (_c = kdfOpts.argon2Iterations) != null ? _c : 3; const para = (_d = kdfOpts.argon2Parallelism) != null ? _d : 1; const len = (_e = kdfOpts.argon2HashLen) != null ? _e : 32; return `argon2id:m${memKB}:t${iters}:p${para}:l${len}:${password}`; } async function deriveKey(password, salt, kdfOpts) { var _a; if ((kdfOpts == null ? void 0 : kdfOpts.type) === "Argon2id") { return deriveKeyArgon2id(password, salt, kdfOpts); } const keyMaterial = await getKeyMaterial(password); const iterations = (_a = kdfOpts == null ? void 0 : kdfOpts.pbkdf2Iterations) != null ? _a : 6e5; return deriveKeyPBKDF2(keyMaterial, salt, iterations); } async function deriveKeyDirect(password, salt, kdfOpts) { return deriveKey(password, salt, kdfOpts); } async function getOrDeriveKey(password, salt, keyCache, kdfOpts) { const cachePass = makeCachePasswordTag(password, kdfOpts); const cached = await keyCache.get(cachePass, salt); if (cached) return cached; const key = await deriveKeyDirect(password, salt, kdfOpts); await keyCache.set(cachePass, salt, key); return key; } function bufferToBase64(buffer) { if (buffer instanceof ArrayBuffer) { return Buffer.from(new Uint8Array(buffer)).toString("base64"); } return Buffer.from(new Uint8Array(buffer.buffer, buffer.byteOffset, buffer.byteLength)).toString("base64"); } function base64ToBuffer(base64) { const buf = Buffer.from(base64, "base64"); return Uint8Array.from(buf); } var init_aesWithPassword = __esm({ "src/encryption/aesWithPassword.ts"() { } }); // src/encryption/eccWithPassword.ts var eccWithPassword_exports = {}; __export(eccWithPassword_exports, { decryptWithPassword: () => decryptWithPassword2, encryptWithPassword: () => encryptWithPassword2 }); async function encryptWithPassword2(password, text, keyCache, kdfOpts) { const encoder = new TextEncoder(); const salt = crypto.getRandomValues(new Uint8Array(16)); const iv = crypto.getRandomValues(new Uint8Array(12)); const keyPair = await crypto.subtle.generateKey( { name: "ECDH", namedCurve: "P-256" }, true, ["deriveKey", "deriveBits"] ); const key = keyCache ? await getOrDeriveKey2(password, salt, keyCache, kdfOpts) : await deriveKeyDirect2(password, salt, kdfOpts); const encrypted = await crypto.subtle.encrypt( { name: "AES-GCM", iv }, key, encoder.encode(text) ); const publicKey = await crypto.subtle.exportKey("raw", keyPair.publicKey); const ephemeralKeyPair = await crypto.subtle.generateKey( { name: "ECDH", namedCurve: "P-256" }, true, ["deriveKey", "deriveBits"] ); const ephemeralPublicKey = await crypto.subtle.exportKey("raw", ephemeralKeyPair.publicKey); return { salt: bufferToBase642(salt), iv: bufferToBase642(iv), data: bufferToBase642(encrypted), publicKey: bufferToBase642(publicKey), ephemeralPublicKey: bufferToBase642(ephemeralPublicKey) }; } async function decryptWithPassword2(password, saltB64, ivB64, dataB64, publicKeyB64, ephemeralPublicKeyB64, keyCache, kdfOpts) { const decoder = new TextDecoder(); const salt = base64ToBuffer2(saltB64); const iv = base64ToBuffer2(ivB64); const data = base64ToBuffer2(dataB64); const key = keyCache ? await getOrDeriveKey2(password, salt, keyCache, kdfOpts) : await deriveKeyDirect2(password, salt, kdfOpts); const decrypted = await crypto.subtle.decrypt( { name: "AES-GCM", iv }, key, data ); return decoder.decode(decrypted); } async function getKeyMaterial2(password) { const enc = new TextEncoder(); return crypto.subtle.importKey( "raw", enc.encode(password), { name: "PBKDF2" }, false, ["deriveBits", "deriveKey"] ); } async function deriveKeyPBKDF22(keyMaterial, salt, iterations) { return crypto.subtle.deriveKey( { name: "PBKDF2", salt, iterations, hash: "SHA-256" }, keyMaterial, { name: "AES-GCM", length: 256 }, false, ["encrypt", "decrypt"] ); } async function deriveKeyArgon2id2(password, salt, kdfOpts) { var _a, _b, _c, _d, _e, _f, _g; const memKB = (_a = kdfOpts == null ? void 0 : kdfOpts.argon2MemoryKB) != null ? _a : 65536; const iters = (_b = kdfOpts == null ? void 0 : kdfOpts.argon2Iterations) != null ? _b : 3; const para = (_c = kdfOpts == null ? void 0 : kdfOpts.argon2Parallelism) != null ? _c : 1; const hashLen = (_d = kdfOpts == null ? void 0 : kdfOpts.argon2HashLen) != null ? _d : 32; try { const argon2 = await Promise.resolve().then(() => __toESM(require_argon23())); const res = await argon2.hash({ pass: password, salt, type: (_f = (_e = argon2.ArgonType) == null ? void 0 : _e.Argon2id) != null ? _f : 2, mem: memKB, time: iters, parallelism: para, hashLen, distPath: void 0 }); const keyBytes = new Uint8Array(res.hash); return crypto.subtle.importKey( "raw", keyBytes, { name: "AES-GCM", length: 256 }, false, ["encrypt", "decrypt"] ); } catch (e) { const keyMaterial = await getKeyMaterial2(password); const iterations = (_g = kdfOpts == null ? void 0 : kdfOpts.pbkdf2Iterations) != null ? _g : 6e5; return deriveKeyPBKDF22(keyMaterial, salt, iterations); } } function makeCachePasswordTag2(password, kdfOpts) { var _a, _b, _c, _d, _e; if (!kdfOpts || kdfOpts.type === "PBKDF2") { const it = (_a = kdfOpts == null ? void 0 : kdfOpts.pbkdf2Iterations) != null ? _a : 6e5; return `pbkdf2:${it}:${password}`; } const memKB = (_b = kdfOpts.argon2MemoryKB) != null ? _b : 65536; const iters = (_c = kdfOpts.argon2Iterations) != null ? _c : 3; const para = (_d = kdfOpts.argon2Parallelism) != null ? _d : 1; const len = (_e = kdfOpts.argon2HashLen) != null ? _e : 32; return `argon2id:m${memKB}:t${iters}:p${para}:l${len}:${password}`; } async function deriveKey2(password, salt, kdfOpts) { var _a; if ((kdfOpts == null ? void 0 : kdfOpts.type) === "Argon2id") { return deriveKeyArgon2id2(password, salt, kdfOpts); } const keyMaterial = await getKeyMaterial2(password); const iterations = (_a = kdfOpts == null ? void 0 : kdfOpts.pbkdf2Iterations) != null ? _a : 6e5; return deriveKeyPBKDF22(keyMaterial, salt, iterations); } async function deriveKeyDirect2(password, salt, kdfOpts) { return deriveKey2(password, salt, kdfOpts); } async function getOrDeriveKey2(password, salt, keyCache, kdfOpts) { const cacheTag = makeCachePasswordTag2(password, kdfOpts); const cached = await keyCache.get(cacheTag, salt); if (cached) return cached; const key = await deriveKeyDirect2(password, salt, kdfOpts); await keyCache.set(cacheTag, salt, key); return key; } function bufferToBase642(buffer) { if (buffer instanceof ArrayBuffer) { return Buffer.from(new Uint8Array(buffer)).toString("base64"); } return Buffer.from(new Uint8Array(buffer.buffer, buffer.byteOffset, buffer.byteLength)).toString("base64"); } function base64ToBuffer2(base64) { const buf = Buffer.from(base64, "base64"); return Uint8Array.from(buf); } var init_eccWithPassword = __esm({ "src/encryption/eccWithPassword.ts"() { } }); // src/encryption/keyCache.ts var keyCache_exports = {}; __export(keyCache_exports, { KeyCache: () => KeyCache }); var KeyCache; var init_keyCache = __esm({ "src/encryption/keyCache.ts"() { KeyCache = class { constructor(ttlMinutes = 5, maxSize = 10) { this.cache = /* @__PURE__ */ new Map(); this.TTL = ttlMinutes * 60 * 1e3; this.MAX_CACHE_SIZE = maxSize; } /** * Generate cache key from password and salt * Uses SHA-256 hash to avoid storing password in plain text */ async generateCacheKey(password, salt) { const encoder = new TextEncoder(); const passwordData = encoder.encode(password); const combined = new Uint8Array(passwordData.length + salt.length); combined.set(passwordData); combined.set(salt, passwordData.length); const hashBuffer = await crypto.subtle.digest("SHA-256", combined); const hashArray = Array.from(new Uint8Array(hashBuffer)); const hashHex = hashArray.map((b) => b.toString(16).padStart(2, "0")).join(""); return hashHex; } /** * Get cached key or return null if not found/expired */ async get(password, salt) { const cacheKey = await this.generateCacheKey(password, salt); const cached = this.cache.get(cacheKey); if (!cached) { return null; } const now = Date.now(); if (now - cached.timestamp > this.TTL) { this.cache.delete(cacheKey); return null; } const saltBase64 = Buffer.from(salt).toString("base64"); if (cached.saltBase64 !== saltBase64) { this.cache.delete(cacheKey); return null; } return cached.key; } /** * Store key in cache */ async set(password, salt, key) { const cacheKey = await this.generateCacheKey(password, salt); const saltBase64 = Buffer.from(salt).toString("base64"); if (this.cache.size >= this.MAX_CACHE_SIZE) { const firstKey = this.cache.keys().next().value; if (firstKey) { this.cache.delete(firstKey); } } this.cache.set(cacheKey, { key, timestamp: Date.now(), saltBase64 }); } /** * Clear all cached keys * Call this when plugin unloads or user logs out */ clear() { this.cache.clear(); } /** * Remove expired entries */ cleanup() { const now = Date.now(); const keysToDelete = []; this.cache.forEach((value, key) => { if (now - value.timestamp > this.TTL) { keysToDelete.push(key); } }); keysToDelete.forEach((key) => this.cache.delete(key)); } /** * Get cache statistics */ getStats() { return { size: this.cache.size, maxSize: this.MAX_CACHE_SIZE, ttl: this.TTL }; } }; } }); // src/types/fileData.ts var _FileDataHelper, FileDataHelper, JsonFileEncoding; var init_fileData = __esm({ "src/types/fileData.ts"() { _FileDataHelper = class { static async encrypt(password, text, type, originalExt, hint, algorithm, keyCache, kdfOptions) { const useAlgorithm = algorithm || "AES"; if (useAlgorithm === "ECC") { const eccCrypto = await Promise.resolve().then(() => (init_eccWithPassword(), eccWithPassword_exports)); const { salt, iv, data, publicKey, ephemeralPublicKey } = await eccCrypto.encryptWithPassword(password, text, keyCache || void 0, kdfOptions); let kdfLine = ""; if ((kdfOptions == null ? void 0 : kdfOptions.type) === "Argon2id") { const mem = kdfOptions.argon2MemoryKB || 65536; const iters = kdfOptions.argon2Iterations || 3; const para = kdfOptions.argon2Parallelism || 1; const len = kdfOptions.argon2HashLen || 32; kdfLine = `KDF:Argon2id:${mem}:${iters}:${para}:${len}`; } else { kdfLine = `KDF:PBKDF2:${(kdfOptions == null ? void 0 : kdfOptions.pbkdf2Iterations) || 6e5}`; } return { version: _FileDataHelper.DEFAULT_VERSION, encodedData: `SALT:${salt} IV:${iv} DATA:${data} PUBLIC_KEY:${publicKey} EPHEMERAL_PUBLIC_KEY:${ephemeralPublicKey} ${kdfLine}`, type, originalExt, hint, algorithm: "ECC", kdf: (kdfOptions == null ? void 0 : kdfOptions.type) || "PBKDF2", iterations: (kdfOptions == null ? void 0 : kdfOptions.type) === "PBKDF2" ? (kdfOptions == null ? void 0 : kdfOptions.pbkdf2Iterations) || 6e5 : void 0, argon2MemoryKB: (kdfOptions == null ? void 0 : kdfOptions.type) === "Argon2id" ? (kdfOptions == null ? void 0 : kdfOptions.argon2MemoryKB) || 65536 : void 0, argon2Iterations: (kdfOptions == null ? void 0 : kdfOptions.type) === "Argon2id" ? (kdfOptions == null ? void 0 : kdfOptions.argon2Iterations) || 3 : void 0, argon2Parallelism: (kdfOptions == null ? void 0 : kdfOptions.type) === "Argon2id" ? (kdfOptions == null ? void 0 : kdfOptions.argon2Parallelism) || 1 : void 0, argon2HashLen: (kdfOptions == null ? void 0 : kdfOptions.type) === "Argon2id" ? (kdfOptions == null ? void 0 : kdfOptions.argon2HashLen) || 32 : void 0 }; } else { const aesCrypto = await Promise.resolve().then(() => (init_aesWithPassword(), aesWithPassword_exports)); const { salt, iv, data } = await aesCrypto.encryptWithPassword(password, text, keyCache || void 0, kdfOptions); let kdfLine = ""; if ((kdfOptions == null ? void 0 : kdfOptions.type) === "Argon2id") { const mem = kdfOptions.argon2MemoryKB || 65536; const iters = kdfOptions.argon2Iterations || 3; const para = kdfOptions.argon2Parallelism || 1; const len = kdfOptions.argon2HashLen || 32; kdfLine = `KDF:Argon2id:${mem}:${iters}:${para}:${len}`; } else { kdfLine = `KDF:PBKDF2:${(kdfOptions == null ? void 0 : kdfOptions.pbkdf2Iterations) || 6e5}`; } return { version: _FileDataHelper.DEFAULT_VERSION, encodedData: `SALT:${salt} IV:${iv} DATA:${data} ${kdfLine}`, type, originalExt, hint, algorithm: "AES", kdf: (kdfOptions == null ? void 0 : kdfOptions.type) || "PBKDF2", iterations: (kdfOptions == null ? void 0 : kdfOptions.type) === "PBKDF2" ? (kdfOptions == null ? void 0 : kdfOptions.pbkdf2Iterations) || 6e5 : void 0, argon2MemoryKB: (kdfOptions == null ? void 0 : kdfOptions.type) === "Argon2id" ? (kdfOptions == null ? void 0 : kdfOptions.argon2MemoryKB) || 65536 : void 0, argon2Iterations: (kdfOptions == null ? void 0 : kdfOptions.type) === "Argon2id" ? (kdfOptions == null ? void 0 : kdfOptions.argon2Iterations) || 3 : void 0, argon2Parallelism: (kdfOptions == null ? void 0 : kdfOptions.type) === "Argon2id" ? (kdfOptions == null ? void 0 : kdfOptions.argon2Parallelism) || 1 : void 0, argon2HashLen: (kdfOptions == null ? void 0 : kdfOptions.type) === "Argon2id" ? (kdfOptions == null ? void 0 : kdfOptions.argon2HashLen) || 32 : void 0 }; } } static async decrypt(data, password, keyCache, kdfOptions) { var _a, _b, _c, _d, _e, _f; if (!data.encodedData) { return ""; } const parts = data.encodedData.split("\n"); if (parts.length < 3) { console.error(`Invalid encoded data format: expected at least 3 parts, got ${parts.length}`); throw new Error(`Invalid encoded data format: expected at least 3 parts, got ${parts.length}`); } const salt = ((_a = parts.find((p) => p.startsWith("SALT:"))) == null ? void 0 : _a.replace("SALT:", "")) || ""; const iv = ((_b = parts.find((p) => p.startsWith("IV:"))) == null ? void 0 : _b.replace("IV:", "")) || ""; const encData = ((_c = parts.find((p) => p.startsWith("DATA:"))) == null ? void 0 : _c.replace("DATA:", "")) || ""; const kdfLine = (_d = parts.find((p) => p.startsWith("KDF:"))) == null ? void 0 : _d.replace("KDF:", ""); let kdfType = (kdfOptions == null ? void 0 : kdfOptions.type) || "PBKDF2"; let iterations = 6e5; let argon2Opts; if (kdfLine) { const segs = kdfLine.split(":"); const parsedType = segs[0].toUpperCase() === "ARGON2ID" ? "Argon2id" : "PBKDF2"; if (!kdfOptions) { kdfType = parsedType; } if (kdfType === "PBKDF2" && segs[1]) { const n = parseInt(segs[1], 10); if (!isNaN(n) && n > 0) iterations = n; } else if (kdfType === "Argon2id" && segs.length >= 5) { argon2Opts = { memoryKB: parseInt(segs[1], 10) || 65536, iterations: parseInt(segs[2], 10) || 3, parallelism: parseInt(segs[3], 10) || 1, hashLen: parseInt(segs[4], 10) || 32 }; } } else if (data.kdf) { if (!kdfOptions) { kdfType = data.kdf; if (kdfType === "PBKDF2" && data.iterations && data.iterations > 0) { iterations = data.iterations; } else if (kdfType === "Argon2id") { argon2Opts = { memoryKB: data.argon2MemoryKB || 65536, iterations: data.argon2Iterations || 3, parallelism: data.argon2Parallelism || 1, hashLen: data.argon2HashLen || 32 }; } } } if (!salt || !iv || !encData) { console.error(`Missing encryption data: salt=${!!salt}, iv=${!!iv}, data=${!!encData}`); throw new Error(`Missing encryption data: salt=${!!salt}, iv=${!!iv}, data=${!!encData}`); } const finalKdfOptions = kdfOptions || (kdfType === "Argon2id" && argon2Opts ? { type: "Argon2id", argon2MemoryKB: argon2Opts.memoryKB, argon2Iterations: argon2Opts.iterations, argon2Parallelism: argon2Opts.parallelism, argon2HashLen: argon2Opts.hashLen } : { type: "PBKDF2", pbkdf2Iterations: iterations }); const publicKey = ((_e = parts.find((p) => p.startsWith("PUBLIC_KEY:"))) == null ? void 0 : _e.replace("PUBLIC_KEY:", "")) || ""; const ephemeralPublicKey = ((_f = parts.find((p) => p.startsWith("EPHEMERAL_PUBLIC_KEY:"))) == null ? void 0 : _f.replace("EPHEMERAL_PUBLIC_KEY:", "")) || ""; const hasKdfInfo = !!kdfLine || !!data.kdf; const tryPbkdf2Fallbacks = async (decryptFn) => { const fallbackIterations = [1e5, 25e4, 5e4, 5e5]; for (const it of fallbackIterations) { if (finalKdfOptions.type === "PBKDF2" && finalKdfOptions.pbkdf2Iterations === it) continue; try { return await decryptFn(it); } catch (e) { } } throw new Error("Decryption failed for all PBKDF2 fallbacks"); }; if (publicKey || data.algorithm === "ECC") { const eccCrypto = await Promise.resolve().then(() => (init_eccWithPassword(), eccWithPassword_exports)); try { const result = await eccCrypto.decryptWithPassword( password, salt, iv, encData, publicKey || "", ephemeralPublicKey, keyCache || void 0, finalKdfOptions ); return result; } catch (error) { console.error("ECC decryption failed:", error); if (!hasKdfInfo && finalKdfOptions.type === "PBKDF2") { try { return await tryPbkdf2Fallbacks(async (iters) => eccCrypto.decryptWithPassword( password, salt, iv, encData, publicKey || "", ephemeralPublicKey, keyCache || void 0, { type: "PBKDF2", pbkdf2Iterations: iters } )); } catch (e) { } } throw error; } } else { const aesCrypto = await Promise.resolve().then(() => (init_aesWithPassword(), aesWithPassword_exports)); try { const result = await aesCrypto.decryptWithPassword(password, salt, iv, encData, keyCache || void 0, finalKdfOptions); return result; } catch (error) { console.error("AES decryption failed:", error); if (!hasKdfInfo && finalKdfOptions.type === "PBKDF2") { try { return await tryPbkdf2Fallbacks(async (iters) => aesCrypto.decryptWithPassword( password, salt, iv, encData, keyCache || void 0, { type: "PBKDF2", pbkdf2Iterations: iters } )); } catch (e) { } } throw error; } } } }; FileDataHelper = _FileDataHelper; FileDataHelper.DEFAULT_VERSION = "2.0"; JsonFileEncoding = class { static encode(data) { return JSON.stringify(data, null, 2); } static decode(text) { var _a, _b, _c, _d, _e, _f, _g, _h, _i; if (!text) { return { version: FileDataHelper.DEFAULT_VERSION, encodedData: "", type: "temporary", originalExt: "md" }; } if (text.startsWith("%%ENC")) { const lines = text.split("\n"); const typeMatch = (_a = lines.find((line) => line.startsWith("TYPE:"))) == null ? void 0 : _a.substring(5); const originalExtMatch = (_b = lines.find((line) => line.startsWith("ORIGINAL_EXT:"))) == null ? void 0 : _b.substring(13); const saltMatch = (_c = lines.find((line) => line.startsWith("SALT:"))) == null ? void 0 : _c.substring(5); const ivMatch = (_d = lines.find((line) => line.startsWith("IV:"))) == null ? void 0 : _d.substring(3); const dataMatch = (_e = lines.find((line) => line.startsWith("DATA:"))) == null ? void 0 : _e.substring(5); const hintMatch = (_f = lines.find((line) => line.startsWith("HINT:"))) == null ? void 0 : _f.substring(5); const kdfMatch = (_g = lines.find((line) => line.startsWith("KDF:"))) == null ? void 0 : _g.substring(4); const publicKeyMatch = (_h = lines.find((line) => line.startsWith("PUBLIC_KEY:"))) == null ? void 0 : _h.substring(11); const ephemeralPublicKeyMatch = (_i = lines.find((line) => line.startsWith("EPHEMERAL_PUBLIC_KEY:"))) == null ? void 0 : _i.substring(21); if (!saltMatch || !ivMatch || !dataMatch) { throw new Error("Invalid file format"); } const encodedLines = [ `SALT:${saltMatch}`, `IV:${ivMatch}`, `DATA:${dataMatch}`, ...publicKeyMatch ? [`PUBLIC_KEY:${publicKeyMatch}`] : [], ...ephemeralPublicKeyMatch ? [`EPHEMERAL_PUBLIC_KEY:${ephemeralPublicKeyMatch}`] : [], ...kdfMatch ? [`KDF:${kdfMatch}`] : [] ]; return { version: FileDataHelper.DEFAULT_VERSION, encodedData: encodedLines.join("\n"), type: typeMatch || "temporary", originalExt: originalExtMatch || "md", hint: hintMatch, algorithm: publicKeyMatch ? "ECC" : "AES" }; } try { return JSON.parse(text); } catch (error) { throw new Error("Invalid file format"); } } static isEncoded(text) { if (text.startsWith("%%ENC")) { return true; } try { const data = JSON.parse(text); return data.version !== void 0 && data.encodedData !== void 0; } catch (e) { return false; } } }; } }); // src/utils/attachmentHelper.ts var attachmentHelper_exports = {}; __export(attachmentHelper_exports, { AttachmentHelper: () => AttachmentHelper }); var import_obsidian4, AttachmentHelper; var init_attachmentHelper = __esm({ "src/utils/attachmentHelper.ts"() { import_obsidian4 = require("obsidian"); init_fileData(); init_aesWithPassword(); AttachmentHelper = class { /** * Normalize a link by removing query/hash, trimming whitespace and angle brackets, * and attempting URL decoding. */ static normalizeLink(raw) { if (!raw) return raw; let link = raw.trim(); if (link.startsWith("<") && link.endsWith(">")) { link = link.substring(1, link.length - 1).trim(); } link = link.split("|")[0]; const hashIdx = link.indexOf("#"); if (hashIdx !== -1) link = link.substring(0, hashIdx); const qIdx = link.indexOf("?"); if (qIdx !== -1) link = link.substring(0, qIdx); try { link = decodeURI(link); } catch (e) { } link = link.replace(/\\/g, "/"); link = link.replace(/\/\/+/, "/"); return link.trim(); } /** * Extract all attachment links from markdown content * Supports: ![[image.png]], ![](image.png), [[document.pdf]] */ static extractAttachmentLinks(content) { const attachments = []; const wikiLinkRegex = /!?\[\[([^\]]+?)\]\]/g; let match; while ((match = wikiLinkRegex.exec(content)) !== null) { const link = this.normalizeLink(match[1]); if (this.isAttachment(link) || link.endsWith(".eccfile") || link.endsWith(".peccfile")) { attachments.push(link); } } const mdImageRegex = /!\[([^\]]*)\]\(([^)]+)\)/g; while ((match = mdImageRegex.exec(content)) !== null) { const link = this.normalizeLink(match[2]); if (this.isAttachment(link) || link.endsWith(".eccfile") || link.endsWith(".peccfile")) { attachments.push(link); } } return [...new Set(attachments)]; } /** * Check if a link points to an attachment (non-markdown file) */ static isAttachment(link) { var _a; const ext = (_a = link.split(".").pop()) == null ? void 0 : _a.toLowerCase(); if (!ext) return false; const attachmentExts = [ "png", "jpg", "jpeg", "gif", "bmp", "svg", "webp", // Images "pdf", "doc", "docx", "xls", "xlsx", "ppt", "pptx", // Documents "mp3", "mp4", "wav", "avi", "mov", // Media "zip", "rar", "7z", // Archives "json", "xml", "csv", "txt" // Data files ]; return attachmentExts.includes(ext); } /** * Resolve attachment path relative to a note */ static resolveAttachmentPath(vault, notePath, attachmentLink) { const cleanLink = this.normalizeLink(attachmentLink); let file = vault.getAbstractFileByPath(cleanLink); if (file instanceof import_obsidian4.TFile) return file; const noteDir = notePath.substring(0, notePath.lastIndexOf("/")); const relativePath = noteDir ? `${noteDir}/${cleanLink}` : cleanLink; file = vault.getAbstractFileByPath(relativePath); if (file instanceof import_obsidian4.TFile) return file; const commonFolders = ["attachments", "assets", "files", "images"]; for (const folder of commonFolders) { file = vault.getAbstractFileByPath(`${folder}/${cleanLink}`); if (file instanceof import_obsidian4.TFile) return file; } const base = cleanLink.split("/").pop(); if (base) { for (const folder of commonFolders) { const candidate = vault.getAbstractFileByPath(`${folder}/${base}`); if (candidate instanceof import_obsidian4.TFile) return candidate; } } return null; } /** * Encrypt a binary file (image, PDF, etc.) */ static async encryptBinaryFile(vault, file, password, type, kdfOpts) { const arrayBuffer = await vault.readBinary(file); const { salt, iv, data } = await encryptBytesWithPassword(password, arrayBuffer, void 0, kdfOpts); let kdfLine = ""; if ((kdfOpts == null ? void 0 : kdfOpts.type) === "Argon2id") { const mem = kdfOpts.argon2MemoryKB || 65536; const iters = kdfOpts.argon2Iterations || 3; const para = kdfOpts.argon2Parallelism || 1; const len = kdfOpts.argon2HashLen || 32; kdfLine = `KDF:Argon2id:${mem}:${iters}:${para}:${len}`; } else { kdfLine = `KDF:PBKDF2:${(kdfOpts == null ? void 0 : kdfOpts.pbkdf2Iterations) || 6e5}`; } const encodedData = `SALT:${salt} IV:${iv} DATA:${data} FORMAT:BIN ${kdfLine}`; const encryptedData = { version: FileDataHelper.DEFAULT_VERSION, encodedData, type, originalExt: file.extension, algorithm: "AES", kdf: (kdfOpts == null ? void 0 : kdfOpts.type) || "PBKDF2", iterations: (kdfOpts == null ? void 0 : kdfOpts.type) === "PBKDF2" ? (kdfOpts == null ? void 0 : kdfOpts.pbkdf2Iterations) || 6e5 : void 0, argon2MemoryKB: (kdfOpts == null ? void 0 : kdfOpts.type) === "Argon2id" ? (kdfOpts == null ? void 0 : kdfOpts.argon2MemoryKB) || 65536 : void 0, argon2Iterations: (kdfOpts == null ? void 0 : kdfOpts.type) === "Argon2id" ? (kdfOpts == null ? void 0 : kdfOpts.argon2Iterations) || 3 : void 0, argon2Parallelism: (kdfOpts == null ? void 0 : kdfOpts.type) === "Argon2id" ? (kdfOpts == null ? void 0 : kdfOpts.argon2Parallelism) || 1 : void 0, argon2HashLen: (kdfOpts == null ? void 0 : kdfOpts.type) === "Argon2id" ? (kdfOpts == null ? void 0 : kdfOpts.argon2HashLen) || 32 : void 0 }; const encryptedExt = type === "permanent" ? "peccfile" : "eccfile"; const encryptedPath = `${file.path}.${encryptedExt}`; const encryptedContent = JSON.stringify(encryptedData, null, 2); const existingFile = vault.getAbstractFileByPath(encryptedPath); if (existingFile instanceof import_obsidian4.TFile) { await vault.modify(existingFile, encryptedContent); } else { await vault.create(encryptedPath, encryptedContent); } await vault.delete(file); return encryptedPath; } /** * Decrypt a binary file */ static async decryptBinaryFile(vault, encryptedFile, password, kdfOpts) { var _a, _b, _c, _d, _e; const content = await vault.read(encryptedFile); const encryptedData = JSON.parse(content); const parts = encryptedData.encodedData.split("\n"); const salt = ((_a = parts.find((p) => p.startsWith("SALT:"))) == null ? void 0 : _a.slice(5)) || ""; const iv = ((_b = parts.find((p) => p.startsWith("IV:"))) == null ? void 0 : _b.slice(3)) || ""; const data = ((_c = parts.find((p) => p.startsWith("DATA:"))) == null ? void 0 : _c.slice(5)) || ""; const format = ((_d = parts.find((p) => p.startsWith("FORMAT:"))) == null ? void 0 : _d.slice(7).toUpperCase()) || ""; if (!salt || !iv || !data) return null; let finalKdfOpts = kdfOpts; const kdfLine = (_e = parts.find((p) => p.startsWith("KDF:"))) == null ? void 0 : _e.slice(4); if (!finalKdfOpts) { if (kdfLine) { const segs = kdfLine.split(":"); if (segs[0].toUpperCase() === "ARGON2ID" && segs.length >= 5) { finalKdfOpts = { type: "Argon2id", argon2MemoryKB: parseInt(segs[1], 10) || 65536, argon2Iterations: parseInt(segs[2], 10) || 3, argon2Parallelism: parseInt(segs[3], 10) || 1, argon2HashLen: parseInt(segs[4], 10) || 32 }; } else if (segs[0].toUpperCase() === "PBKDF2") { finalKdfOpts = { type: "PBKDF2", pbkdf2Iterations: parseInt(segs[1], 10) || 6e5 }; } } else if (encryptedData.kdf) { if (encryptedData.kdf === "Argon2id") { finalKdfOpts = { type: "Argon2id", argon2MemoryKB: encryptedData.argon2MemoryKB || 65536, argon2Iterations: encryptedData.argon2Iterations || 3, argon2Parallelism: encryptedData.argon2Parallelism || 1, argon2HashLen: encryptedData.argon2HashLen || 32 }; } else { finalKdfOpts = { type: "PBKDF2", pbkdf2Iterations: encryptedData.iterations || 6e5 }; } } } const tryLegacyPbkdf2 = async (fn) => { const fallbackIterations = [1e5, 25e4, 5e4, 5e5]; for (const it of fallbackIterations) { if (finalKdfOpts && finalKdfOpts.type === "PBKDF2" && finalKdfOpts.pbkdf2Iterations === it) continue; try { return await fn(it); } catch (e) { } } return null; }; const isPbkdf2NoMetadata = !kdfLine && !encryptedData.kdf && (!finalKdfOpts || finalKdfOpts.type === "PBKDF2"); let arrayBuffer; if (format === "BIN") { try { const bytes = await decryptBytesWithPassword(password, salt, iv, data, void 0, finalKdfOpts); arrayBuffer = bytes.buffer.slice(bytes.byteOffset, bytes.byteOffset + bytes.byteLength); } catch (err) { if (isPbkdf2NoMetadata) { const fallback = await tryLegacyPbkdf2(async (iters) => { const bytes = await decryptBytesWithPassword( password, salt, iv, data, void 0, { type: "PBKDF2", pbkdf2Iterations: iters } ); return bytes; }); if (fallback) { arrayBuffer = fallback.buffer.slice(fallback.byteOffset, fallback.byteOffset + fallback.byteLength); } else { throw err; } } else { throw err; } } } else { try { const base64 = await FileDataHelper.decrypt(encryptedData, password, void 0, finalKdfOpts); if (!base64) return null; arrayBuffer = this.base64ToArrayBuffer(base64); } catch (err) { if (isPbkdf2NoMetadata) { const fallback = await tryLegacyPbkdf2(async (iters) => FileDataHelper.decrypt( encryptedData, password, void 0, { type: "PBKDF2", pbkdf2Iterations: iters } )); if (fallback) { arrayBuffer = this.base64ToArrayBuffer(fallback); } else { throw err; } } else { throw err; } } } const originalPath = encryptedFile.path.replace(/\.eccfile$/, "").replace(/\.peccfile$/, ""); const existingFile = vault.getAbstractFileByPath(originalPath); if (existingFile instanceof import_obsidian4.TFile) { await vault.modifyBinary(existingFile, arrayBuffer); } else { await vault.createBinary(originalPath, arrayBuffer); } if (encryptedData.type === "temporary") { await vault.delete(encryptedFile); } return vault.getAbstractFileByPath(originalPath); } /** * Update attachment links in note content * @param content - The note content * @param encryptionType - "encrypt" to add .eccirian/.peccirian, "decrypt" to remove * @param mode - "temporary" or "permanent" */ static updateAttachmentLinks(content, encryptionType, mode) { const ext = mode === "permanent" ? "peccfile" : "eccfile"; if (encryptionType === "encrypt") { content = content.replace( /(!?\[\[)([^\]]+?)(\]\])/g, (match, prefix, filename, suffix) => { if (this.isAttachment(filename)) { return `${prefix}${filename}.${ext}${suffix}`; } return match; } ); content = content.replace( /(!)\[([^\]]*)\]\(([^)]+?)\)/g, (match, exclaim, alt, path) => { if (this.isAttachment(path)) { return `${exclaim}[${alt}](${path}.${ext})`; } return match; } ); } else { content = content.replace( new RegExp(`(!?\\[\\[[^\\]]+?)\\.${ext}(\\]\\])`, "g"), "$1$2" ); content = content.replace( new RegExp(`(!\\[[^\\]]*\\]\\([^)]+?)\\.${ext}(\\))`, "g"), "$1$2" ); } return content; } /** * Convert ArrayBuffer to Base64 */ static arrayBufferToBase64(buffer) { return Buffer.from(new Uint8Array(buffer)).toString("base64"); } /** * Convert Base64 to ArrayBuffer */ static base64ToArrayBuffer(base64) { const buf = Buffer.from(base64, "base64"); return buf.buffer.slice(buf.byteOffset, buf.byteOffset + buf.byteLength); } }; } }); // src/main.ts var main_exports = {}; __export(main_exports, { default: () => EccEncryptPlugin }); module.exports = __toCommonJS(main_exports); var import_obsidian9 = require("obsidian"); init_aesWithPassword(); init_eccWithPassword(); // src/ui/passwordModal.ts var import_obsidian = require("obsidian"); var PasswordModal = class extends import_obsidian.Modal { constructor(app, onSuccess, onCancel, defaultMode = "temporary", isDecrypt = false, requirePasswordConfirmation = false, showHint = false, defaultEncryptionMethod = "AES", plugin, showModeSelection = true, existingHint) { super(app); this.password = ""; this.confirmPassword = ""; this.hint = ""; this.encryptionMode = "temporary"; this.encryptionMethod = "AES"; this.isSubmitted = false; this.isDecrypt = false; this.requirePasswordConfirmation = false; this.showHint = false; this.showModeSelection = true; this.onSuccess = onSuccess; this.onCancel = onCancel; this.encryptionMode = defaultMode; this.isDecrypt = isDecrypt; this.requirePasswordConfirmation = requirePasswordConfirmation; this.showHint = showHint; this.encryptionMethod = defaultEncryptionMethod; this.plugin = plugin; this.showModeSelection = showModeSelection; this.existingHint = existingHint; } onOpen() { const { contentEl } = this; contentEl.empty(); new import_obsidian.Setting(contentEl).setName("Eccirian Encryption").setHeading(); if (this.isDecrypt && this.existingHint) { const hintContainer = contentEl.createDiv({ cls: "setting-item" }); const hintInfo = hintContainer.createDiv({ cls: "setting-item-info" }); hintInfo.createDiv({ cls: "setting-item-name", text: "Password Hint" }); const hintControl = hintContainer.createDiv({ cls: "setting-item-control" }); const hintCode = hintControl.createEl("code", { text: this.existingHint, cls: "eccirian-hint-code" }); hintCode.style.padding = "4px 8px"; hintCode.style.backgroundColor = "var(--background-modifier-border)"; hintCode.style.borderRadius = "4px"; hintCode.style.fontSize = "0.9em"; } if (!this.isDecrypt) { new import_obsidian.Setting(contentEl).setName("Encryption Method").setDesc("Choose encryption method").addDropdown( (drop) => drop.addOption("AES", "AES-256 (Symmetric)").addOption("ECC", "ECC+AES (Asymmetric)").setValue(this.encryptionMethod).onChange((value) => { this.encryptionMethod = value; }) ); if (this.showModeSelection) { new import_obsidian.Setting(contentEl).setName("Encryption Mode").setDesc("Choose encryption mode").addDropdown( (drop) => drop.addOption("temporary", "Temporary (Single Use)").addOption("permanent", "Permanent (Repeatable)").setValue(this.encryptionMode).onChange((value) => { this.encryptionMode = value; }) ); } } new import_obsidian.Setting(contentEl).setName("Password").addText((text) => { text.setPlaceholder("Enter password").setValue(this.password).onChange((value) => { this.password = value; }); text.inputEl.type = "password"; text.inputEl.addEventListener("keypress", (e) => { if (e.key === "Enter" && this.password) { if (!this.requirePasswordConfirmation || this.confirmPassword) { this.submit(); } } }); }); if (!this.isDecrypt && this.requirePasswordConfirmation) { new import_obsidian.Setting(contentEl).setName("Confirm Password").addText((text) => { text.setPlaceholder("Confirm password").setValue(this.confirmPassword).onChange((value) => { this.confirmPassword = value; }); text.inputEl.type = "password"; text.inputEl.addEventListener("keypress", (e) => { if (e.key === "Enter" && this.password && this.confirmPassword) { this.submit(); } }); }); } if (!this.isDecrypt && this.showHint) { new import_obsidian.Setting(contentEl).setName("Password Hint").setDesc("Optional hint to help remember the password").addText((text) => { text.setPlaceholder("Enter password hint (optional)").setValue(this.hint).onChange((value) => { this.hint = value; }); }); } new import_obsidian.Setting(contentEl).addButton((button) => { button.setButtonText("Confirm").setCta().onClick(() => { this.submit(); }); }).addButton((button) => { button.setButtonText("Cancel").onClick(() => { this.close(); this.onCancel(); }); }); } async submit() { if (!this.password) { if (this.plugin.settings.showNotice) new import_obsidian.Notice("Please enter a password"); return; } if (!this.isDecrypt && this.requirePasswordConfirmation) { if (!this.confirmPassword) { if (this.plugin.settings.showNotice) new import_obsidian.Notice("Please confirm your password"); return; } if (this.password !== this.confirmPassword) { if (this.plugin.settings.showNotice) new import_obsidian.Notice("Passwords do not match"); return; } } if (!this.isDecrypt) { const extension = this.encryptionMode === "permanent" ? "peccirian" : "eccirian"; this.plugin.settings.fileExtension = extension; await this.plugin.saveSettings(); } this.isSubmitted = true; this.close(); this.onSuccess(this.password, this.encryptionMethod, this.hint, this.encryptionMode); } onClose() { const { contentEl } = this; contentEl.empty(); if (!this.isSubmitted) { this.onCancel(); } } }; // src/ui/settingsTab.ts var import_obsidian2 = require("obsidian"); // src/i18n/index.ts var en = { // Sections settingsSectionGeneral: "General", settingsSectionUI: "UI", settingsSectionSecurity: "Security", settingsSectionAdvanced: "Advanced", // Settings settingsDefaultEncryptionMethod: "Default Encryption Method", settingsDefaultEncryptionMethodDesc: "Choose a method to use to encrypt the note content", settingsDefaultEncryptionMode: "Default Encryption Mode", settingsDefaultEncryptionModeDesc: "Choose the default encryption mode", settingsIconStyle: "Icon Style", settingsIconStyleDesc: "Choose the icon style for the lock page", settingsRequirePasswordConfirmation: "Require Password Confirmation", settingsRequirePasswordConfirmationDesc: "When enabled, requires entering password twice when encrypting", settingsShowToggleExtensionButton: "Show Toggle Extension Button", settingsShowToggleExtensionButtonDesc: "Show the button to toggle between .md and .eccirian extensions", settingsShowNotices: "Show Notices", settingsShowNoticesDesc: "Show notification messages when encrypting/decrypting files", settingsShowHint: "Show Password Hint", settingsShowHintDesc: "Allow adding password hints when encrypting", settingsEncryptAttachments: "Encrypt Attachments", settingsEncryptAttachmentsDesc: "Automatically encrypt attachments (images, PDFs, etc.) linked in encrypted notes", settingsFileExtension: "Encrypted File Extension", settingsFileExtensionDesc: "Default extension for encrypted notes (without dot)", settingsKdfType: "Key Derivation Function", settingsKdfTypeDesc: "Choose PBKDF2 or Argon2id", settingsPbkdf2Iterations: "PBKDF2 Iterations", settingsPbkdf2IterationsDesc: "Higher is safer but slower. (Default: 600,000, WASP 2023)", settingsArgon2Memory: "Argon2id Memory (KB)", settingsArgon2MemoryDesc: "Memory cost in KiB. Higher = more secure, slower.", settingsArgon2Iterations: "Argon2id Iterations", settingsArgon2IterationsDesc: "Time cost (number of iterations).", settingsArgon2Parallelism: "Argon2id Parallelism", settingsArgon2ParallelismDesc: "Degree of parallelism (lanes/threads); higher uses more CPU.", settingsArgon2HashLen: "Argon2id Hash Length (bytes)", settingsArgon2HashLenDesc: "Derived key length. 32 is typical for AES-256.", kdfOptionPBKDF2: "PBKDF2", kdfOptionArgon2id: "Argon2id", // Advanced settings settingsEnableKeyCache: "Enable Key Cache", settingsEnableKeyCacheDesc: "Cache derived keys in memory to speed up repeated operations (recommended)", settingsKeyCacheTTL: "Key Cache TTL (Time to Live)", settingsKeyCacheTTLDesc: "How long to keep cached keys in memory (minutes, default: 5)", settingsKeyCacheMaxSize: "Key Cache Max Size", settingsKeyCacheMaxSizeDesc: "Maximum number of keys to cache simultaneously (default: 10)", settingsClearKeyCache: "Clear Key Cache", settingsClearKeyCacheDesc: "Manually clear all cached keys from memory", buttonClearCache: "Clear Cache", messageKeyCacheCleared: "Key cache cleared", messageKeyCacheNotEnabled: "Key cache is not enabled", // Encryption methods encryptionMethodAES: "AES-256 (Symmetric)", encryptionMethodECC: "ECC+AES (Asymmetric)", // Encryption modes encryptionModeTemporary: "Temporary (Single Use)", encryptionModePermanent: "Permanent (Repeatable)", // Icon styles iconStyleLock: "\u{1F512} Lock", iconStyleShield: "\u{1F6E1}\uFE0F Shield", iconStyleKey: "\u{1F511} Key", iconStylePadlock: "\u{1F510} Padlock", // Messages messageFileEncrypted: "File encrypted", messageFileDecrypted: "File decrypted", messageDecryptionFailed: "Decryption failed", messageInvalidFileFormat: "Invalid file format", messagePasswordIncorrect: "Password may be incorrect", messageUnexpectedError: "Unexpected error", messageFileDoesNotExist: "File does not exist", messageNotEncryptedFile: "This is not an encrypted file, restored as normal file", messagePleaseOpenFileFirst: "Please open a file first", messageOnlyMdCanBeConverted: "Only .md files can be converted to .eccirian", messageFileAlreadyDecrypted: "This file is already decrypted", messageConversionSuccess: "File converted to markdown successfully", messageConversionFailed: "Conversion failed", messagePleaseDecryptFirst: "Please decrypt the file first", messageFailedToGetCurrentFile: "Failed to get current file", messageFolderEncrypted: "Folder encrypted", messageFolderDecrypted: "Folder decrypted", commandEncryptFolder: "Encrypt Folder", // Lock view lockViewMessage: "is encrypted. Click below to unlock.", lockViewUnlockButton: "Unlock", // File conflict dialogs dialogFileAlreadyExists: "File Already Exists", dialogFileAlreadyExistsDesc1: 'A file named "{filename}" already exists.', dialogFileAlreadyExistsDesc2: "Decrypting will overwrite the existing file.", dialogFileAlreadyExistsDesc3: "Do you want to continue?", buttonCancel: "Cancel", buttonContinue: "Continue" }; var zhCN = { // Sections settingsSectionGeneral: "\u5E38\u89C4", settingsSectionUI: "\u754C\u9762", settingsSectionSecurity: "\u5B89\u5168", settingsSectionAdvanced: "\u9AD8\u7EA7", // Settings settingsDefaultEncryptionMethod: "\u9ED8\u8BA4\u52A0\u5BC6\u7B97\u6CD5", settingsDefaultEncryptionMethodDesc: "\u9009\u62E9\u4F7F\u7528\u54EA\u79CD\u65B9\u5F0F\u52A0\u5BC6\u7B14\u8BB0\u5185\u5BB9", settingsDefaultEncryptionMode: "\u9ED8\u8BA4\u52A0\u5BC6\u6A21\u5F0F", settingsDefaultEncryptionModeDesc: "\u9009\u62E9\u9ED8\u8BA4\u52A0\u5BC6\u6A21\u5F0F", settingsIconStyle: "\u56FE\u6807\u6837\u5F0F", settingsIconStyleDesc: "\u9009\u62E9\u9501\u5B9A\u9875\u9762\u56FE\u6807\u6837\u5F0F", settingsRequirePasswordConfirmation: "\u9700\u8981\u5BC6\u7801\u786E\u8BA4", settingsRequirePasswordConfirmationDesc: "\u542F\u7528\u540E\uFF0C\u52A0\u5BC6\u65F6\u9700\u8981\u8F93\u5165\u4E24\u6B21\u5BC6\u7801", settingsShowToggleExtensionButton: "\u663E\u793A\u6269\u5C55\u540D\u8F6C\u6362\u6309\u94AE", settingsShowToggleExtensionButtonDesc: "\u663E\u793A\u5728 .md \u548C .eccirian \u6269\u5C55\u540D\u4E4B\u95F4\u5207\u6362\u7684\u6309\u94AE", settingsShowNotices: "\u663E\u793A\u901A\u77E5", settingsShowNoticesDesc: "\u52A0\u5BC6/\u89E3\u5BC6\u6587\u4EF6\u65F6\u663E\u793A\u901A\u77E5\u6D88\u606F", settingsShowHint: "\u663E\u793A\u5BC6\u7801\u63D0\u793A", settingsShowHintDesc: "\u52A0\u5BC6\u65F6\u5141\u8BB8\u6DFB\u52A0\u5BC6\u7801\u63D0\u793A", settingsEncryptAttachments: "\u52A0\u5BC6\u9644\u4EF6", settingsEncryptAttachmentsDesc: "\u81EA\u52A8\u52A0\u5BC6\u7B14\u8BB0\u4E2D\u94FE\u63A5\u7684\u9644\u4EF6\uFF08\u56FE\u7247\u3001PDF\u7B49\uFF09", settingsFileExtension: "\u52A0\u5BC6\u6587\u4EF6\u6269\u5C55\u540D", settingsFileExtensionDesc: "\u52A0\u5BC6\u7B14\u8BB0\u7684\u9ED8\u8BA4\u6269\u5C55\u540D\uFF08\u4E0D\u542B\u70B9\uFF09", settingsKdfType: "\u5BC6\u94A5\u6D3E\u751F\u51FD\u6570", settingsKdfTypeDesc: "\u9009\u62E9 PBKDF2 \u6216 Argon2id", settingsPbkdf2Iterations: "PBKDF2 \u8FED\u4EE3\u6B21\u6570", settingsPbkdf2IterationsDesc: "\u6B21\u6570\u8D8A\u9AD8\u8D8A\u5B89\u5168\u4F46\u8D8A\u6162\uFF0C\u63A8\u8350\uFF1A100,000", settingsArgon2Memory: "Argon2id \u5185\u5B58 (KB)", settingsArgon2MemoryDesc: "\u5185\u5B58\u6D88\u8017\uFF08KiB\uFF09\uFF0C\u66F4\u9AD8\u66F4\u5B89\u5168\u4F46\u66F4\u6162\u3002", settingsArgon2Iterations: "Argon2id \u8FED\u4EE3\u6B21\u6570", settingsArgon2IterationsDesc: "\u65F6\u95F4\u6210\u672C\uFF08\u8FED\u4EE3\u6B21\u6570\uFF09", settingsArgon2Parallelism: "Argon2id \u5E76\u884C\u5EA6", settingsArgon2ParallelismDesc: "\u5E76\u884C\u901A\u9053\u6570\u91CF\uFF08lanes/\u7EBF\u7A0B\uFF09\uFF0C\u66F4\u9AD8\u4F1A\u5360\u7528\u66F4\u591A CPU", settingsArgon2HashLen: "Argon2id \u54C8\u5E0C\u957F\u5EA6 (\u5B57\u8282)", settingsArgon2HashLenDesc: "\u6D3E\u751F\u5BC6\u94A5\u957F\u5EA6\uFF0C32 \u9002\u7528\u4E8E AES-256", kdfOptionPBKDF2: "PBKDF2", kdfOptionArgon2id: "Argon2id", // Advanced settings settingsEnableKeyCache: "\u542F\u7528\u5BC6\u94A5\u7F13\u5B58", settingsEnableKeyCacheDesc: "\u5728\u5185\u5B58\u4E2D\u7F13\u5B58\u6D3E\u751F\u5BC6\u94A5\u4EE5\u52A0\u901F\u91CD\u590D\u64CD\u4F5C\uFF08\u63A8\u8350\uFF09", settingsKeyCacheTTL: "\u5BC6\u94A5\u7F13\u5B58\u5B58\u6D3B\u65F6\u95F4", settingsKeyCacheTTLDesc: "\u5728\u5185\u5B58\u4E2D\u4FDD\u7559\u7F13\u5B58\u5BC6\u94A5\u7684\u65F6\u957F\uFF08\u5206\u949F\uFF0C\u9ED8\u8BA45\uFF09", settingsKeyCacheMaxSize: "\u5BC6\u94A5\u7F13\u5B58\u6700\u5927\u6570\u91CF", settingsKeyCacheMaxSizeDesc: "\u540C\u65F6\u7F13\u5B58\u7684\u6700\u5927\u5BC6\u94A5\u6570\u91CF\uFF08\u9ED8\u8BA410\uFF09", settingsClearKeyCache: "\u6E05\u9664\u5BC6\u94A5\u7F13\u5B58", settingsClearKeyCacheDesc: "\u624B\u52A8\u6E05\u9664\u5185\u5B58\u4E2D\u7684\u6240\u6709\u7F13\u5B58\u5BC6\u94A5", buttonClearCache: "\u6E05\u9664\u7F13\u5B58", messageKeyCacheCleared: "\u5BC6\u94A5\u7F13\u5B58\u5DF2\u6E05\u9664", messageKeyCacheNotEnabled: "\u5BC6\u94A5\u7F13\u5B58\u672A\u542F\u7528", // Encryption methods encryptionMethodAES: "AES-256\uFF08\u5BF9\u79F0\u52A0\u5BC6\uFF09", encryptionMethodECC: "ECC+AES\uFF08\u975E\u5BF9\u79F0\u52A0\u5BC6\uFF09", // Encryption modes encryptionModeTemporary: "\u4E34\u65F6\uFF08\u5355\u6B21\u4F7F\u7528\uFF09", encryptionModePermanent: "\u6C38\u4E45\uFF08\u91CD\u590D\u4F7F\u7528\uFF09", // Icon styles iconStyleLock: "\u{1F512} \u9501", iconStyleShield: "\u{1F6E1}\uFE0F \u76FE\u724C", iconStyleKey: "\u{1F511} \u94A5\u5319", iconStylePadlock: "\u{1F510} \u6302\u9501", // Messages messageFileEncrypted: "\u6587\u4EF6\u5DF2\u52A0\u5BC6", messageFileDecrypted: "\u6587\u4EF6\u5DF2\u89E3\u5BC6", messageDecryptionFailed: "\u89E3\u5BC6\u5931\u8D25", messageInvalidFileFormat: "\u65E0\u6548\u7684\u6587\u4EF6\u683C\u5F0F", messagePasswordIncorrect: "\u5BC6\u7801\u53EF\u80FD\u4E0D\u6B63\u786E", messageUnexpectedError: "\u610F\u5916\u9519\u8BEF", messageFileDoesNotExist: "\u6587\u4EF6\u4E0D\u5B58\u5728", messageNotEncryptedFile: "\u8FD9\u4E0D\u662F\u52A0\u5BC6\u6587\u4EF6\uFF0C\u5DF2\u6062\u590D\u4E3A\u666E\u901A\u6587\u4EF6", messagePleaseOpenFileFirst: "\u8BF7\u5148\u6253\u5F00\u4E00\u4E2A\u6587\u4EF6", messageOnlyMdCanBeConverted: "\u53EA\u6709 .md \u6587\u4EF6\u53EF\u4EE5\u8F6C\u6362\u4E3A .eccirian", messageFileAlreadyDecrypted: "\u6B64\u6587\u4EF6\u5DF2\u89E3\u5BC6", messageConversionSuccess: "\u6587\u4EF6\u5DF2\u6210\u529F\u8F6C\u6362\u4E3A markdown", messageConversionFailed: "\u8F6C\u6362\u5931\u8D25", messagePleaseDecryptFirst: "\u8BF7\u5148\u89E3\u5BC6\u6587\u4EF6", messageFailedToGetCurrentFile: "\u65E0\u6CD5\u83B7\u53D6\u5F53\u524D\u6587\u4EF6", messageFolderEncrypted: "\u6587\u4EF6\u5939\u5DF2\u52A0\u5BC6", messageFolderDecrypted: "\u6587\u4EF6\u5939\u5DF2\u89E3\u5BC6", commandEncryptFolder: "\u52A0\u5BC6\u6587\u4EF6\u5939", // Lock view lockViewMessage: "\u5DF2\u52A0\u5BC6\u3002\u70B9\u51FB\u4E0B\u65B9\u89E3\u9501\u3002", lockViewUnlockButton: "\u89E3\u9501", // File conflict dialogs dialogFileAlreadyExists: "\u6587\u4EF6\u5DF2\u5B58\u5728", dialogFileAlreadyExistsDesc1: '\u540D\u4E3A "{filename}" \u7684\u6587\u4EF6\u5DF2\u5B58\u5728\u3002', dialogFileAlreadyExistsDesc2: "\u89E3\u5BC6\u5C06\u8986\u76D6\u73B0\u6709\u6587\u4EF6\u3002", dialogFileAlreadyExistsDesc3: "\u8981\u7EE7\u7EED\u5417\uFF1F", buttonCancel: "\u53D6\u6D88", buttonContinue: "\u7EE7\u7EED" }; var locales = { "en": en, "zh": zhCN, "zh-CN": zhCN, "zh-cn": zhCN }; function getI18n(locale) { if (locales[locale]) { return locales[locale]; } const langCode = locale.split("-")[0]; if (locales[langCode]) { return locales[langCode]; } return en; } function t(key, locale) { const i18n = getI18n(locale); return i18n[key] || key; } // src/ui/settingsTab.ts var ConfirmModal = class extends import_obsidian2.Modal { constructor(app) { super(app); } onOpen() { const { contentEl } = this; new import_obsidian2.Setting(contentEl).setName("Visit GitHub Repository?").setHeading(); contentEl.createEl("p", { text: "Do you want to visit the GitHub repository?" }); new import_obsidian2.Setting(contentEl).addButton((button) => { button.setButtonText("Cancel").onClick(() => { this.close(); }); }).addButton((button) => { button.setButtonText("Confirm").setCta().onClick(() => { window.open("https://github.com/Enthalpiex/eccirian-encrypt", "_blank"); this.close(); }); }); } onClose() { const { contentEl } = this; contentEl.empty(); } }; var EccEncryptSettingTab = class extends import_obsidian2.PluginSettingTab { constructor(app, plugin) { super(app, plugin); this.plugin = plugin; } display() { const { containerEl } = this; containerEl.empty(); const locale = import_obsidian2.moment.locale(); const logoContainer = containerEl.createDiv("eccirian-logo-container"); logoContainer.style.textAlign = "center"; logoContainer.style.marginBottom = "2rem"; logoContainer.style.cursor = "pointer"; logoContainer.style.width = "100%"; const logo = logoContainer.createEl("img", { attr: { src: "", alt: "Eccirian Logo" } }); logo.style.width = "100%"; logo.style.height = "auto"; logo.style.objectFit = "cover"; logoContainer.addEventListener("click", () => { new ConfirmModal(this.app).open(); }); new import_obsidian2.Setting(containerEl).setName(t("settingsSectionGeneral", locale)).setHeading(); new import_obsidian2.Setting(containerEl).setName(t("settingsDefaultEncryptionMethod", locale)).setDesc(t("settingsDefaultEncryptionMethodDesc", locale)).addDropdown( (drop) => drop.addOption("AES", t("encryptionMethodAES", locale)).addOption("ECC", t("encryptionMethodECC", locale)).setValue(this.plugin.settings.encryptionMethod).onChange(async (value) => { this.plugin.settings.encryptionMethod = value; await this.plugin.saveSettings(); }) ); new import_obsidian2.Setting(containerEl).setName(t("settingsDefaultEncryptionMode", locale)).setDesc(t("settingsDefaultEncryptionModeDesc", locale)).addDropdown( (drop) => drop.addOption("temporary", t("encryptionModeTemporary", locale)).addOption("permanent", t("encryptionModePermanent", locale)).setValue(this.plugin.settings.defaultEncryptionMode).onChange(async (value) => { this.plugin.settings.defaultEncryptionMode = value; await this.plugin.saveSettings(); }) ); new import_obsidian2.Setting(containerEl).setName(t("settingsSectionUI", locale)).setHeading(); new import_obsidian2.Setting(containerEl).setName(t("settingsIconStyle", locale)).setDesc(t("settingsIconStyleDesc", locale)).addDropdown( (drop) => drop.addOption("lock", t("iconStyleLock", locale)).addOption("shield", t("iconStyleShield", locale)).addOption("key", t("iconStyleKey", locale)).addOption("padlock", t("iconStylePadlock", locale)).setValue(this.plugin.settings.iconStyle).onChange(async (value) => { this.plugin.settings.iconStyle = value; await this.plugin.saveSettings(); }) ); new import_obsidian2.Setting(containerEl).setName(t("settingsRequirePasswordConfirmation", locale)).setDesc(t("settingsRequirePasswordConfirmationDesc", locale)).addToggle( (toggle) => toggle.setValue(this.plugin.settings.requirePasswordConfirmation).onChange(async (value) => { this.plugin.settings.requirePasswordConfirmation = value; await this.plugin.saveSettings(); }) ); new import_obsidian2.Setting(containerEl).setName(t("settingsShowToggleExtensionButton", locale)).setDesc(t("settingsShowToggleExtensionButtonDesc", locale)).addToggle( (toggle) => toggle.setValue(this.plugin.settings.showToggleExtensionButton).onChange(async (value) => { this.plugin.settings.showToggleExtensionButton = value; await this.plugin.saveSettings(); this.plugin.updateToggleExtensionButton(); }) ); new import_obsidian2.Setting(containerEl).setName(t("settingsShowNotices", locale)).setDesc(t("settingsShowNoticesDesc", locale)).addToggle( (toggle) => toggle.setValue(this.plugin.settings.showNotice).onChange(async (value) => { this.plugin.settings.showNotice = value; await this.plugin.saveSettings(); }) ); new import_obsidian2.Setting(containerEl).setName(t("settingsShowHint", locale)).setDesc(t("settingsShowHintDesc", locale)).addToggle( (toggle) => toggle.setValue(this.plugin.settings.showHint).onChange(async (value) => { this.plugin.settings.showHint = value; await this.plugin.saveSettings(); }) ); new import_obsidian2.Setting(containerEl).setName(t("settingsSectionSecurity", locale)).setHeading(); new import_obsidian2.Setting(containerEl).setName(t("settingsEncryptAttachments", locale)).setDesc(t("settingsEncryptAttachmentsDesc", locale)).addToggle( (toggle) => toggle.setValue(this.plugin.settings.encryptAttachments).onChange(async (value) => { this.plugin.settings.encryptAttachments = value; await this.plugin.saveSettings(); }) ); new import_obsidian2.Setting(containerEl).setName(t("settingsKdfType", locale)).setDesc(t("settingsKdfTypeDesc", locale)).addDropdown( (drop) => drop.addOption("PBKDF2", t("kdfOptionPBKDF2", locale)).addOption("Argon2id", t("kdfOptionArgon2id", locale)).setValue(this.plugin.settings.kdfType).onChange(async (value) => { const prevScroll = containerEl.scrollTop; this.plugin.settings.kdfType = value; await this.plugin.saveSettings(); this.display(); setTimeout(() => { containerEl.scrollTop = prevScroll; }, 0); }) ); if (this.plugin.settings.kdfType === "PBKDF2") { new import_obsidian2.Setting(containerEl).setName(t("settingsPbkdf2Iterations", locale)).setDesc(t("settingsPbkdf2IterationsDesc", locale)).addDropdown( (drop) => drop.addOption("100000", "100,000 (Fast)").addOption("600000", "600,000 (Balanced)").addOption("1000000", "1,000,000 (Strong)").setValue(String(this.plugin.settings.pbkdf2Iterations)).onChange(async (value) => { const v = parseInt(value, 10); this.plugin.settings.pbkdf2Iterations = v; await this.plugin.saveSettings(); }) ); } if (this.plugin.settings.kdfType === "Argon2id") { new import_obsidian2.Setting(containerEl).setName(t("settingsArgon2Memory", locale)).setDesc(t("settingsArgon2MemoryDesc", locale)).addText( (text) => text.setPlaceholder("65536").setValue(String(this.plugin.settings.argon2MemoryKB)).onChange(async (value) => { const v = Math.max(8192, parseInt(value || "65536", 10) || 65536); this.plugin.settings.argon2MemoryKB = v; await this.plugin.saveSettings(); }) ); new import_obsidian2.Setting(containerEl).setName(t("settingsArgon2Iterations", locale)).setDesc(t("settingsArgon2IterationsDesc", locale)).addDropdown( (drop) => drop.addOption("1", "1 (Fast)").addOption("3", "3 (Balanced)").addOption("5", "5 (Strong)").setValue(String(this.plugin.settings.argon2Iterations)).onChange(async (value) => { const v = parseInt(value, 10) || 3; this.plugin.settings.argon2Iterations = v; await this.plugin.saveSettings(); }) ); new import_obsidian2.Setting(containerEl).setName(t("settingsArgon2Parallelism", locale)).setDesc(t("settingsArgon2ParallelismDesc", locale)).addDropdown( (drop) => drop.addOption("1", "1").addOption("2", "2").addOption("4", "4").setValue(String(this.plugin.settings.argon2Parallelism)).onChange(async (value) => { const v = parseInt(value, 10) || 1; this.plugin.settings.argon2Parallelism = v; await this.plugin.saveSettings(); }) ); new import_obsidian2.Setting(containerEl).setName(t("settingsArgon2HashLen", locale)).setDesc(t("settingsArgon2HashLenDesc", locale)).addDropdown( (drop) => drop.addOption("16", "16").addOption("32", "32 (AES-256)").addOption("64", "64").setValue(String(this.plugin.settings.argon2HashLen)).onChange(async (value) => { const v = parseInt(value, 10) || 32; this.plugin.settings.argon2HashLen = v; await this.plugin.saveSettings(); }) ); } new import_obsidian2.Setting(containerEl).setName(t("settingsSectionAdvanced", locale)).setHeading(); new import_obsidian2.Setting(containerEl).setName(t("settingsEnableKeyCache", locale)).setDesc(t("settingsEnableKeyCacheDesc", locale)).addToggle( (toggle) => toggle.setValue(this.plugin.settings.enableKeyCache).onChange(async (value) => { this.plugin.settings.enableKeyCache = value; await this.plugin.saveSettings(); if (value && !this.plugin.keyCache) { const { KeyCache: KeyCache2 } = await Promise.resolve().then(() => (init_keyCache(), keyCache_exports)); this.plugin.keyCache = new KeyCache2( this.plugin.settings.keyCacheTTL, this.plugin.settings.keyCacheMaxSize ); } else if (!value && this.plugin.keyCache) { this.plugin.keyCache.clear(); this.plugin.keyCache = null; } this.display(); }) ); if (this.plugin.settings.enableKeyCache) { new import_obsidian2.Setting(containerEl).setName(t("settingsKeyCacheTTL", locale)).setDesc(t("settingsKeyCacheTTLDesc", locale)).addSlider( (slider) => slider.setLimits(1, 30, 1).setValue(this.plugin.settings.keyCacheTTL).setDynamicTooltip().onChange(async (value) => { this.plugin.settings.keyCacheTTL = value; await this.plugin.saveSettings(); if (this.plugin.keyCache) { const { KeyCache: KeyCache2 } = await Promise.resolve().then(() => (init_keyCache(), keyCache_exports)); const oldCache = this.plugin.keyCache; oldCache.clear(); this.plugin.keyCache = new KeyCache2(value, this.plugin.settings.keyCacheMaxSize); } }) ); new import_obsidian2.Setting(containerEl).setName(t("settingsKeyCacheMaxSize", locale)).setDesc(t("settingsKeyCacheMaxSizeDesc", locale)).addSlider( (slider) => slider.setLimits(5, 50, 5).setValue(this.plugin.settings.keyCacheMaxSize).setDynamicTooltip().onChange(async (value) => { this.plugin.settings.keyCacheMaxSize = value; await this.plugin.saveSettings(); if (this.plugin.keyCache) { const { KeyCache: KeyCache2 } = await Promise.resolve().then(() => (init_keyCache(), keyCache_exports)); const oldCache = this.plugin.keyCache; oldCache.clear(); this.plugin.keyCache = new KeyCache2(this.plugin.settings.keyCacheTTL, value); } }) ); new import_obsidian2.Setting(containerEl).setName(t("settingsClearKeyCache", locale)).setDesc(t("settingsClearKeyCacheDesc", locale)).addButton( (button) => button.setButtonText(t("buttonClearCache", locale)).setWarning().onClick(() => { if (this.plugin.settings.showNotice) { if (this.plugin.keyCache) { this.plugin.keyCache.clear(); new import_obsidian2.Notice(t("messageKeyCacheCleared", locale)); } else { new import_obsidian2.Notice(t("messageKeyCacheNotEnabled", locale)); } } }) ); } } }; // src/settings.ts var DEFAULT_SETTINGS = { encryptionMethod: "ECC", defaultEncryptionMode: "temporary", iconStyle: "lock", requirePasswordConfirmation: true, showToggleExtensionButton: true, showNotice: false, showHint: true, fileExtension: "eccirian", encryptAttachments: true, kdfType: "PBKDF2", pbkdf2Iterations: 6e5, argon2MemoryKB: 65536, argon2Iterations: 3, argon2Parallelism: 1, argon2HashLen: 32, // Advanced defaults enableKeyCache: true, keyCacheTTL: 5, keyCacheMaxSize: 10 }; // src/utils/fileHelper.ts var import_obsidian3 = require("obsidian"); async function changeFileExtension(vault, file, newExt) { const pathParts = file.path.split("/"); const fileName = pathParts.pop(); const baseName = fileName.replace(/\.[^/.]+$/, ""); const newPath = [...pathParts, `${baseName}.${newExt}`].join("/"); const result = await vault.rename(file, newPath); const newFile = vault.getAbstractFileByPath(newPath); if (!(newFile instanceof import_obsidian3.TFile)) { throw new Error(`Expected a file at path ${newPath} but got a folder or null`); } return newFile; } // src/view/eccirianView.ts var import_obsidian6 = require("obsidian"); var import_obsidian7 = require("obsidian"); init_aesWithPassword(); init_eccWithPassword(); init_attachmentHelper(); // src/utils/folderHelper.ts var import_obsidian5 = require("obsidian"); var FolderHelper = class { /** * Pack folder contents into a JSON structure */ static async packFolder(vault, folder) { const folderData = { isFolder: true, folderName: folder.name, folderPath: folder.path, files: [], subfolders: [] }; const files = this.getAllFilesInFolder(folder); for (const file of files) { const content = await vault.read(file); folderData.files.push({ path: file.path.replace(folder.path + "/", ""), // Relative path content, extension: file.extension }); } return JSON.stringify(folderData); } /** * Unpack folder from JSON structure */ static async unpackFolder(vault, folderData, targetPath) { const data = JSON.parse(folderData); if (!data.isFolder) { throw new Error("Not a folder encryption data"); } const folderPath = targetPath || data.folderPath; const existingFolder = vault.getAbstractFileByPath(folderPath); if (!existingFolder) { await vault.createFolder(folderPath); } for (const fileData of data.files) { const filePath = `${folderPath}/${fileData.path}`; const parentPath = filePath.substring(0, filePath.lastIndexOf("/")); if (parentPath && !vault.getAbstractFileByPath(parentPath)) { await vault.createFolder(parentPath); } const existingFile = vault.getAbstractFileByPath(filePath); if (existingFile instanceof import_obsidian5.TFile) { await vault.modify(existingFile, fileData.content); } else { await vault.create(filePath, fileData.content); } } } /** * Get all files in a folder recursively */ static getAllFilesInFolder(folder) { const files = []; for (const child of folder.children) { if (child instanceof import_obsidian5.TFile) { files.push(child); } else if (child instanceof import_obsidian5.TFolder) { files.push(...this.getAllFilesInFolder(child)); } } return files; } /** * Check if encrypted content is a folder */ static isFolderEncryption(content) { try { const data = JSON.parse(content); return data.isFolder === true; } catch (e) { return false; } } /** * Get folder metadata from encrypted content */ static getFolderMetadata(content) { try { const data = JSON.parse(content); if (!data.isFolder) return null; return { folderName: data.folderName, fileCount: data.files.length, folderPath: data.folderPath }; } catch (e) { return null; } } }; // src/view/eccirianView.ts init_fileData(); var ECCIDIAN_VIEW_TYPE = "eccirian-encrypt-view"; var EccidianView = class extends import_obsidian6.FileView { constructor(leaf, plugin) { super(leaf); this.loadingFile = false; this.displayName = "Encrypted Note"; this.filePath = ""; this.navigation = true; this.plugin = plugin; } updateTitle(newTitle) { this.displayName = newTitle; } getViewType() { return ECCIDIAN_VIEW_TYPE; } getDisplayText() { if (this.file) { return this.file.basename; } return this.displayName; } getIcon() { return "lock"; } async setState(state, result) { await super.setState(state, result); if (state == null ? void 0 : state.file) { if (state.file instanceof import_obsidian7.TFile) { this.file = state.file; this.filePath = state.file.path; this.updateTitle(state.file.basename); } else if (typeof state.file === "string") { this.filePath = state.file; const file = this.app.vault.getAbstractFileByPath(this.filePath); if (file instanceof import_obsidian7.TFile) { this.file = file; this.updateTitle(file.basename); } } } await this.onOpen(); } getState() { return { file: this.filePath, type: ECCIDIAN_VIEW_TYPE }; } async onOpen() { const container = this.containerEl.children[1]; const containerEl = this.containerEl; containerEl.style.backgroundColor = "var(--background-primary)"; containerEl.style.position = "relative"; const contentContainer = container; contentContainer.style.position = "relative"; let isFolder = false; let folderMetadata = null; if (this.file) { try { const content = await this.app.vault.read(this.file); const isFolderMatch = content.match(/IS_FOLDER:(.+)/); isFolder = !!(isFolderMatch && isFolderMatch[1] === "true"); if (isFolder) { } } catch (error) { console.error("Error checking folder status:", error); } } const iconMap = { lock: "\u{1F512}", shield: "\u{1F6E1}\uFE0F", key: "\u{1F511}", padlock: "\u{1F510}" }; const icon = iconMap[this.plugin.settings.iconStyle] || "\u{1F512}"; const displayIcon = isFolder ? "\u{1F4C1}" : icon; const lockView = container.createDiv({ cls: "eccirian-lock-view" }); lockView.style.cssText = ` position: absolute; top: 0; left: 0; right: 0; bottom: 0; height: 100%; width: 100%; display: flex; flex-direction: column; align-items: center; justify-content: center; text-align: center; padding: 2rem; background-color: var(--background-primary); z-index: 1000; pointer-events: auto; `; const contentElements = contentContainer.querySelectorAll("*:not(.eccirian-lock-view)"); contentElements.forEach((el) => { if (el !== lockView && !lockView.contains(el)) { el.style.pointerEvents = "none"; } }); lockView.addEventListener("keydown", (e) => { e.stopPropagation(); e.preventDefault(); }); lockView.addEventListener("keyup", (e) => { e.stopPropagation(); e.preventDefault(); }); lockView.addEventListener("keypress", (e) => { e.stopPropagation(); e.preventDefault(); }); lockView.addEventListener("mousedown", (e) => { if (e.target.tagName !== "BUTTON") { e.stopPropagation(); } }); const lockIcon = lockView.createDiv({ cls: "eccirian-lock-icon" }); lockIcon.style.cssText = ` font-size: 64px; margin-bottom: 1.5rem; opacity: 0.8; transition: transform 0.3s ease; `; lockIcon.setText(displayIcon); const message = lockView.createEl("p", { cls: "eccirian-message" }); message.style.cssText = ` margin-bottom: 2rem; font-size: 1.1rem; color: var(--text-muted); line-height: 1.5; `; const itemType = isFolder ? "folder" : "file"; message.setText(this.file ? `"${this.file.basename}" is an encrypted ${itemType}. Click below to unlock.` : `This file is encrypted. Click below to unlock.`); const unlockBtn = lockView.createEl("button", { cls: "eccirian-unlock-button" }); unlockBtn.style.cssText = ` padding: 0.75rem 1.5rem; font-size: 1rem; cursor: pointer; border-radius: 6px; border: 1px solid var(--background-modifier-border); background-color: var(--background-primary); color: var(--text-normal); transition: all 0.2s ease; box-shadow: 0 2px 4px rgba(0,0,0,0.1); `; unlockBtn.setText("Unlock"); unlockBtn.addEventListener("mouseenter", () => { unlockBtn.style.backgroundColor = "var(--background-secondary)"; unlockBtn.style.transform = "translateY(-1px)"; lockIcon.style.transform = "scale(1.1)"; }); unlockBtn.addEventListener("mouseleave", () => { unlockBtn.style.backgroundColor = "var(--background-primary)"; unlockBtn.style.transform = "translateY(0)"; lockIcon.style.transform = "scale(1)"; }); unlockBtn.addEventListener("click", async () => { if (this.filePath) { const file = this.app.vault.getAbstractFileByPath(this.filePath); if (file instanceof import_obsidian7.TFile) { this.file = file; } } if (!this.file) { if (this.plugin.settings.showNotice) { new import_obsidian7.Notice("File does not exist"); } return; } try { const encContent = await this.app.vault.read(this.file); const looksJson = encContent.trim().startsWith("{") && encContent.includes("encodedData"); if (!encContent.includes("%%ENC") && !looksJson) { if (this.plugin.settings.showNotice) { new import_obsidian7.Notice("This is not an encrypted file"); } const leaf = this.app.workspace.getLeaf(); await leaf.openFile(this.file, { state: { mode: "source" } }); return; } const isFolderMatch0 = encContent.match(/IS_FOLDER:(.+)/); const isFolder0 = isFolderMatch0 && isFolderMatch0[1] === "true"; if (!encContent.includes("%%ENC") && looksJson) { try { const dataObj = JSON.parse(encContent); const originalExt2 = (dataObj.originalExt || "md").toLowerCase(); const encoded = String(dataObj.encodedData || ""); new PasswordModal( this.app, async (password) => { var _a, _b, _c, _d, _e; try { const parts = encoded.split("\n"); const saltLine = parts.find((p) => p.startsWith("SALT:")); const ivLine = parts.find((p) => p.startsWith("IV:")); const dataLine = parts.find((p) => p.startsWith("DATA:")); const fmtLine = parts.find((p) => p.startsWith("FORMAT:")); const kdfLine = (_a = parts.find((p) => p.startsWith("KDF:"))) == null ? void 0 : _a.slice(4); if (!saltLine || !ivLine || !dataLine) { if (this.plugin.settings.showNotice) new import_obsidian7.Notice("Invalid JSON format"); return; } const salt = saltLine.slice(5); const iv = ivLine.slice(3); const data = dataLine.slice(5); const fmt = (fmtLine ? fmtLine.slice(7) : "").toUpperCase(); let kdfOpts; if (kdfLine) { const segs = kdfLine.split(":"); if (segs[0].toUpperCase() === "ARGON2ID" && segs.length >= 5) { kdfOpts = { type: "Argon2id", argon2MemoryKB: parseInt(segs[1], 10) || 65536, argon2Iterations: parseInt(segs[2], 10) || 3, argon2Parallelism: parseInt(segs[3], 10) || 1, argon2HashLen: parseInt(segs[4], 10) || 32 }; } else if (segs[0].toUpperCase() === "PBKDF2") { kdfOpts = { type: "PBKDF2", pbkdf2Iterations: parseInt(segs[1], 10) || 6e5 }; } } else if (dataObj.kdf) { if (dataObj.kdf === "Argon2id") { kdfOpts = { type: "Argon2id", argon2MemoryKB: dataObj.argon2MemoryKB || 65536, argon2Iterations: dataObj.argon2Iterations || 3, argon2Parallelism: dataObj.argon2Parallelism || 1, argon2HashLen: dataObj.argon2HashLen || 32 }; } else { kdfOpts = { type: "PBKDF2", pbkdf2Iterations: dataObj.iterations || 6e5 }; } } if (originalExt2 !== "md") { let arrayBuffer; if (fmt === "BIN") { const { decryptBytesWithPassword: decryptBytesWithPassword2 } = await Promise.resolve().then(() => (init_aesWithPassword(), aesWithPassword_exports)); const bytes = await decryptBytesWithPassword2(password, salt, iv, data, ((_c = (_b = this.plugin) == null ? void 0 : _b.getKeyCache) == null ? void 0 : _c.call(_b)) || void 0, kdfOpts); arrayBuffer = bytes.buffer.slice(bytes.byteOffset, bytes.byteOffset + bytes.byteLength); } else { const txt = await FileDataHelper.decrypt(dataObj, password, ((_e = (_d = this.plugin) == null ? void 0 : _d.getKeyCache) == null ? void 0 : _e.call(_d)) || void 0, kdfOpts); if (!txt) { if (this.plugin.settings.showNotice) new import_obsidian7.Notice("Decryption failed"); return; } const buf = Buffer.from(txt, "base64"); arrayBuffer = buf.buffer.slice(buf.byteOffset, buf.byteOffset + buf.byteLength); } if (!this.file) { if (this.plugin.settings.showNotice) new import_obsidian7.Notice("File not found"); return; } const basePath = this.file.path.replace(/\.eccirian$/, "").replace(/\.peccirian$/, ""); const originalPath = basePath.toLowerCase().endsWith(".md") ? basePath : `${basePath}.md`; const existing = this.app.vault.getAbstractFileByPath(originalPath); if (existing instanceof import_obsidian7.TFile) { const shouldOverwrite = await new Promise((resolve) => { const modal = new import_obsidian7.Modal(this.app); modal.titleEl.setText(this.plugin.i18n.dialogFileAlreadyExists || "File Already Exists"); const desc = modal.contentEl.createDiv(); const filename = originalPath.split("/").pop() || originalPath; desc.createEl("p", { text: (this.plugin.i18n.dialogFileAlreadyExistsDesc1 || 'A file named "{filename}" already exists.').replace("{filename}", filename) }); desc.createEl("p", { text: this.plugin.i18n.dialogFileAlreadyExistsDesc2 || "Decrypting will overwrite the existing file.", cls: "mod-warning" }); desc.createEl("p", { text: this.plugin.i18n.dialogFileAlreadyExistsDesc3 || "Do you want to continue?" }); const buttonContainer = modal.contentEl.createDiv("modal-button-container"); const cancelBtn = buttonContainer.createEl("button", { text: this.plugin.i18n.buttonCancel || "Cancel", cls: "mod-cancel" }); cancelBtn.addEventListener("click", () => { resolve(false); modal.close(); }); const continueBtn = buttonContainer.createEl("button", { text: this.plugin.i18n.buttonContinue || "Continue", cls: "mod-cta mod-warning" }); continueBtn.addEventListener("click", () => { resolve(true); modal.close(); }); modal.open(); }); if (!shouldOverwrite) { return; } await this.app.vault.delete(existing); } if (existing instanceof import_obsidian7.TFile) { await this.app.vault.modifyBinary(existing, arrayBuffer); } else { await this.app.vault.createBinary(originalPath, arrayBuffer); } await this.app.vault.delete(this.file); if (this.plugin.settings.showNotice) new import_obsidian7.Notice("File decrypted"); const restored = this.app.vault.getAbstractFileByPath(originalPath); if (restored instanceof import_obsidian7.TFile) { const leaf = this.app.workspace.getLeaf(); await leaf.openFile(restored); } return; } } catch (e) { if (this.plugin.settings.showNotice) new import_obsidian7.Notice("Decryption failed"); } }, async () => { }, this.plugin.settings.defaultEncryptionMode, true, this.plugin.settings.requirePasswordConfirmation, this.plugin.settings.showHint, this.plugin.settings.encryptionMethod, this.plugin, true, void 0 ).open(); return; } catch (e) { } } if (isFolder0) { const saltMatch2 = encContent.match(/SALT:(.+)/); const ivMatch2 = encContent.match(/IV:(.+)/); const dataMatch2 = encContent.match(/DATA:(.+)/); const typeMatch2 = encContent.match(/TYPE:(.+)/); const hintMatch2 = encContent.match(/HINT:(.+)/); const encryptionType2 = typeMatch2 ? typeMatch2[1] : "temporary"; const hint2 = hintMatch2 ? hintMatch2[1] : void 0; if (!saltMatch2 || !ivMatch2 || !dataMatch2) { if (this.plugin.settings.showNotice) new import_obsidian7.Notice("Decryption failed: Invalid file format"); return; } new PasswordModal( this.app, async (password) => { var _a, _b, _c, _d, _e; try { let decrypted; const publicKeyMatch = encContent.match(/PUBLIC_KEY:(.+)/); const ephemeralPublicKeyMatch = encContent.match(/EPHEMERAL_PUBLIC_KEY:(.+)/); const kdfLine = encContent.match(/KDF:(.+)/); let kdfOpts; if (kdfLine) { const segs = kdfLine[1].split(":"); if (segs[0].toUpperCase() === "ARGON2ID" && segs.length >= 5) { kdfOpts = { type: "Argon2id", argon2MemoryKB: parseInt(segs[1], 10) || 65536, argon2Iterations: parseInt(segs[2], 10) || 3, argon2Parallelism: parseInt(segs[3], 10) || 1, argon2HashLen: parseInt(segs[4], 10) || 32 }; } else if (segs[0].toUpperCase() === "PBKDF2") { kdfOpts = { type: "PBKDF2", pbkdf2Iterations: parseInt(segs[1], 10) || 6e5 }; } } if (publicKeyMatch) { const ephemeralKey = ephemeralPublicKeyMatch ? ephemeralPublicKeyMatch[1] : void 0; decrypted = await decryptWithPassword2( password, saltMatch2[1], ivMatch2[1], dataMatch2[1], publicKeyMatch[1], ephemeralKey, ((_b = (_a = this.plugin) == null ? void 0 : _a.getKeyCache) == null ? void 0 : _b.call(_a)) || void 0, kdfOpts ); } else { decrypted = await decryptWithPassword( password, saltMatch2[1], ivMatch2[1], dataMatch2[1], ((_d = (_c = this.plugin) == null ? void 0 : _c.getKeyCache) == null ? void 0 : _d.call(_c)) || void 0, kdfOpts ); } const originalPathMatch = encContent.match(/ORIGINAL_PATH:(.+)/); const originalPath = originalPathMatch ? originalPathMatch[1] : null; if (!originalPath) { if (this.plugin.settings.showNotice) new import_obsidian7.Notice("Missing ORIGINAL_PATH"); return; } const getAvailableFolderPath = (basePath) => { const lastSlash = basePath.lastIndexOf("/"); const dir = lastSlash >= 0 ? basePath.substring(0, lastSlash) : ""; const name = lastSlash >= 0 ? basePath.substring(lastSlash + 1) : basePath; const buildPath = (n2) => { const suffix = n2 === 0 ? "" : n2 === 1 ? " (Restored)" : ` (Restored ${n2})`; const finalName = `${name}${suffix}`; return dir ? `${dir}/${finalName}` : finalName; }; let n = 0; let candidate = buildPath(n); while (this.app.vault.getAbstractFileByPath(candidate)) { n += 1; candidate = buildPath(n); } return candidate; }; const targetPath = getAvailableFolderPath(originalPath); await FolderHelper.unpackFolder(this.app.vault, decrypted, targetPath); if (this.file) { await this.app.vault.delete(this.file); } if (this.plugin.settings.showNotice) { new import_obsidian7.Notice(`${this.plugin.i18n.messageFolderDecrypted} \u2192 ${targetPath}`); } } catch (error) { console.error("Folder decryption failed:", error); console.error("Salt:", saltMatch2 == null ? void 0 : saltMatch2[1]); console.error("IV:", ivMatch2 == null ? void 0 : ivMatch2[1]); console.error("Data:", ((_e = dataMatch2 == null ? void 0 : dataMatch2[1]) == null ? void 0 : _e.substring(0, 50)) + "..."); console.error("Error details:", error instanceof Error ? error.message : String(error)); if (this.plugin.settings.showNotice) { new import_obsidian7.Notice("Decryption failed: " + (error instanceof Error ? error.message : "Unknown error")); } } }, async () => { }, this.plugin.settings.defaultEncryptionMode, true, this.plugin.settings.requirePasswordConfirmation, this.plugin.settings.showHint, this.plugin.settings.encryptionMethod, this.plugin, true, hint2 ).open(); return; } const mdPath = this.file.path.replace(/\.[^/.]+$/, ".md"); const existingMdFile = this.app.vault.getAbstractFileByPath(mdPath); if (existingMdFile) { const continueDecrypt = await new Promise((resolve) => { const modal = new import_obsidian7.Modal(this.app); modal.titleEl.setText(this.plugin.i18n.dialogFileAlreadyExists || "File Already Exists"); const desc = modal.contentEl.createDiv(); const filename = mdPath.split("/").pop() || mdPath; desc.createEl("p", { text: (this.plugin.i18n.dialogFileAlreadyExistsDesc1 || 'A file named "{filename}" already exists.').replace("{filename}", filename) }); desc.createEl("p", { text: this.plugin.i18n.dialogFileAlreadyExistsDesc2 || "Decrypting will overwrite the existing file.", cls: "mod-warning" }); desc.createEl("p", { text: this.plugin.i18n.dialogFileAlreadyExistsDesc3 || "Do you want to continue?" }); const buttonContainer = modal.contentEl.createDiv("modal-button-container"); const cancelBtn = buttonContainer.createEl("button", { text: this.plugin.i18n.buttonCancel || "Cancel", cls: "mod-cancel" }); cancelBtn.addEventListener("click", () => { resolve(false); modal.close(); }); const continueBtn = buttonContainer.createEl("button", { text: this.plugin.i18n.buttonContinue || "Continue", cls: "mod-cta mod-warning" }); continueBtn.addEventListener("click", () => { resolve(true); modal.close(); }); modal.open(); }); if (!continueDecrypt) { return; } await this.app.vault.delete(existingMdFile); } const mdFile = await changeFileExtension(this.app.vault, this.file, "md"); const content = await this.app.vault.read(mdFile); if (!content.includes("%%ENC")) { if (this.plugin.settings.showNotice) new import_obsidian7.Notice("This is not an encrypted file, restored as normal file"); const leaf = this.app.workspace.getLeaf(); await leaf.openFile(mdFile, { state: { mode: "source" } }); return; } const saltMatch = content.match(/SALT:(.+)/); const ivMatch = content.match(/IV:(.+)/); const dataMatch = content.match(/DATA:(.+)/); const typeMatch = content.match(/TYPE:(.+)/); const isFolderMatch = content.match(/IS_FOLDER:(.+)/); const isFolder2 = isFolderMatch && isFolderMatch[1] === "true"; const originalExtMatch = content.match(/ORIGINAL_EXT:(.+)/); if (!saltMatch || !ivMatch || !dataMatch || !typeMatch) { await changeFileExtension(this.app.vault, mdFile, "eccirian"); if (this.plugin.settings.showNotice) { new import_obsidian7.Notice("Decryption failed: Invalid file format"); } return; } if (!isFolder2 && !originalExtMatch) { await changeFileExtension(this.app.vault, mdFile, "eccirian"); if (this.plugin.settings.showNotice) { new import_obsidian7.Notice("Decryption failed: Invalid file format"); } return; } const encryptionType = typeMatch[1]; const originalExt = originalExtMatch ? originalExtMatch[1] : "md"; const hintMatch = content.match(/HINT:(.+)/); const hint = hintMatch ? hintMatch[1] : void 0; this.plugin.settings.defaultEncryptionMode = encryptionType; await this.plugin.saveSettings(); new PasswordModal( this.app, async (password) => { var _a, _b; try { const dataObj = JsonFileEncoding.decode(content); const decrypted = await FileDataHelper.decrypt(dataObj, password, ((_b = (_a = this.plugin) == null ? void 0 : _a.getKeyCache) == null ? void 0 : _b.call(_a)) || void 0); if (!decrypted) { await changeFileExtension(this.app.vault, mdFile, "eccirian"); if (this.plugin.settings.showNotice) new import_obsidian7.Notice("Decryption failed: Password may be incorrect"); return; } try { const isFolderMatch2 = content.match(/IS_FOLDER:(.+)/); const isFolder3 = isFolderMatch2 && isFolderMatch2[1] === "true"; if (isFolder3) { const originalPathMatch = content.match(/ORIGINAL_PATH:(.+)/); const originalPath2 = originalPathMatch ? originalPathMatch[1] : null; if (originalPath2) { const getAvailableFolderPath = (basePath) => { const lastSlash = basePath.lastIndexOf("/"); const dir = lastSlash >= 0 ? basePath.substring(0, lastSlash) : ""; const name = lastSlash >= 0 ? basePath.substring(lastSlash + 1) : basePath; const buildPath = (n2) => { const suffix = n2 === 0 ? "" : n2 === 1 ? " (Restored)" : ` (Restored ${n2})`; const finalName = `${name}${suffix}`; return dir ? `${dir}/${finalName}` : finalName; }; let n = 0; let candidate = buildPath(n); while (this.app.vault.getAbstractFileByPath(candidate)) { n += 1; candidate = buildPath(n); } return candidate; }; const targetPath = getAvailableFolderPath(originalPath2); await FolderHelper.unpackFolder(this.app.vault, decrypted, targetPath); await this.app.vault.delete(mdFile); if (this.plugin.settings.showNotice) { new import_obsidian7.Notice(`${this.plugin.i18n.messageFolderDecrypted} \u2192 ${targetPath}`); } return; } } let processedText = decrypted.trim() === "$\xB7-\xB7$" ? "" : decrypted; const decodedType = JsonFileEncoding.decode(content).type; if (this.plugin.settings.encryptAttachments && decodedType === "temporary") { const attachmentLinks = AttachmentHelper.extractAttachmentLinks(processedText); await Promise.all( attachmentLinks.map(async (link) => { try { const encryptedAttachment = AttachmentHelper.resolveAttachmentPath( this.app.vault, mdFile.path, link ); if (!encryptedAttachment) { console.warn(`Attachment not found: ${link}`); return; } await AttachmentHelper.decryptBinaryFile( this.app.vault, encryptedAttachment, password ); } catch (error) { console.error(`Failed to decrypt attachment ${link}:`, error); } }) ); processedText = AttachmentHelper.updateAttachmentLinks( processedText, "decrypt", "temporary" ); } await this.app.vault.modify(mdFile, processedText); if (this.plugin.settings.showNotice) new import_obsidian7.Notice("File decrypted"); const leaf = this.app.workspace.getLeaf(); const decoded = JsonFileEncoding.decode(content); const originalExt2 = decoded.originalExt || "md"; const originalPath = mdFile.path.replace(/\.[^/.]+$/, `.${originalExt2}`); const existingOriginalFile = this.app.vault.getAbstractFileByPath(originalPath); if (existingOriginalFile && existingOriginalFile !== mdFile) { const continueRestore = await new Promise((resolve) => { const modal = new import_obsidian7.Modal(this.app); modal.titleEl.setText(this.plugin.i18n.dialogFileAlreadyExists || "File Already Exists"); const desc = modal.contentEl.createDiv(); const filename = originalPath.split("/").pop() || originalPath; desc.createEl("p", { text: (this.plugin.i18n.dialogFileAlreadyExistsDesc1 || 'A file named "{filename}" already exists.').replace("{filename}", filename) }); desc.createEl("p", { text: this.plugin.i18n.dialogFileAlreadyExistsDesc2 || "Decrypting will overwrite the existing file.", cls: "mod-warning" }); desc.createEl("p", { text: this.plugin.i18n.dialogFileAlreadyExistsDesc3 || "Do you want to continue?" }); const buttonContainer = modal.contentEl.createDiv("modal-button-container"); const cancelBtn = buttonContainer.createEl("button", { text: this.plugin.i18n.buttonCancel || "Cancel", cls: "mod-cancel" }); cancelBtn.addEventListener("click", () => { resolve(false); modal.close(); }); const continueBtn = buttonContainer.createEl("button", { text: this.plugin.i18n.buttonContinue || "Continue", cls: "mod-cta mod-warning" }); continueBtn.addEventListener("click", () => { resolve(true); modal.close(); }); modal.open(); }); if (!continueRestore) { await leaf.openFile(mdFile, { state: { mode: "source" } }); return; } await this.app.vault.delete(existingOriginalFile); } const originalFile = await changeFileExtension(this.app.vault, mdFile, originalExt2); await leaf.openFile(originalFile, { state: { mode: "source" } }); } catch (err) { const leaf = this.app.workspace.getLeaf(); await leaf.openFile(mdFile, { state: { mode: "source" } }); return; } } catch (err) { console.error("Decryption error:", err); await changeFileExtension(this.app.vault, mdFile, "eccirian"); if (this.plugin.settings.showNotice) { new import_obsidian7.Notice("Decryption failed: Unexpected error"); } return; } }, async () => { try { await changeFileExtension(this.app.vault, mdFile, "eccirian"); } catch (err) { if (this.plugin.settings.showNotice) { new import_obsidian7.Notice("Failed to restore file extension"); } } }, this.plugin.settings.defaultEncryptionMode, true, this.plugin.settings.requirePasswordConfirmation, this.plugin.settings.showHint, this.plugin.settings.encryptionMethod, this.plugin, true, // Show mode selection hint // Pass the hint to display ).open(); } catch (err) { if (this.plugin.settings.showNotice) new import_obsidian7.Notice("Failed to change file extension"); } }); } async onClose() { await super.onClose(); } }; // src/view/peccirianView.ts var import_obsidian8 = require("obsidian"); init_fileData(); var PECCIDIAN_VIEW_TYPE = "peccidian-encrypt-view"; var PEccidianView = class extends import_obsidian8.MarkdownView { constructor(leaf, plugin) { super(leaf); this.loadingFile = false; this.displayName = "Encrypted Note"; this.password = null; this.encryptedData = null; this.isSavingEnabled = true; this.isLoadingFileInProgress = false; this.isSavingInProgress = false; this.isPasswordModalOpen = false; this.isInitialized = false; this.lastChangeTime = 0; this.changeDebounceTimeout = null; this.memoryKey = null; this.isViewActive = false; this.isSwitchingView = false; this.lockView = null; this.isLockViewVisible = false; this.lastOpenedFilePath = null; this.plugin = plugin; this.filePath = ""; this.generateMemoryKey(); } generateMemoryKey() { const array = new Uint8Array(32); crypto.getRandomValues(array); this.memoryKey = Array.from(array, (byte) => byte.toString(16).padStart(2, "0")).join(""); } encryptForMemory(text) { if (!this.memoryKey) { this.generateMemoryKey(); } const key = this.memoryKey; let result = ""; for (let i = 0; i < text.length; i++) { const charCode = text.charCodeAt(i) ^ key.charCodeAt(i % key.length); result += String.fromCharCode(charCode); } return btoa(result); } decryptFromMemory(encryptedText) { if (!this.memoryKey) return encryptedText; const key = this.memoryKey; const text = atob(encryptedText); let result = ""; for (let i = 0; i < text.length; i++) { const charCode = text.charCodeAt(i) ^ key.charCodeAt(i % key.length); result += String.fromCharCode(charCode); } return result; } setPassword(password) { const encryptedPassword = this.encryptForMemory(password); this.password = encryptedPassword; } getPassword() { if (!this.password) return null; return this.decryptFromMemory(this.password); } getViewType() { return PECCIDIAN_VIEW_TYPE; } updateTitle(newTitle) { this.displayName = newTitle; const titleEl = this.contentEl.querySelector(".inline-title"); if (titleEl) { titleEl.textContent = newTitle; titleEl.setAttribute("data-inline-title", newTitle); } } getDisplayText() { if (this.file) { return this.file.basename; } return this.displayName; } getIcon() { return "lock"; } createLockView() { if (this.lockView) { this.lockView.remove(); this.lockView = null; } const contentEl = this.contentEl; if (contentEl) { contentEl.style.pointerEvents = "none"; } const containerEl = this.containerEl; containerEl.style.backgroundColor = "var(--background-primary)"; containerEl.style.position = "relative"; const container = this.containerEl.children[1]; container.style.position = "relative"; this.lockView = container.createDiv({ cls: "peccirian-lock-view" }); this.lockView.style.cssText = ` position: absolute; top: 0; left: 0; right: 0; bottom: 0; height: 100%; width: 100%; display: flex; flex-direction: column; align-items: center; justify-content: center; text-align: center; padding: 2rem; background-color: var(--background-primary); z-index: 1000; pointer-events: auto; `; this.lockView.addEventListener("keydown", (e) => { e.stopPropagation(); e.preventDefault(); }); this.lockView.addEventListener("keyup", (e) => { e.stopPropagation(); e.preventDefault(); }); this.lockView.addEventListener("keypress", (e) => { e.stopPropagation(); e.preventDefault(); }); this.lockView.addEventListener("mousedown", (e) => { if (e.target.tagName !== "BUTTON") { e.stopPropagation(); } }); const iconMap = { lock: "\u{1F512}", shield: "\u{1F6E1}\uFE0F", key: "\u{1F511}", padlock: "\u{1F510}" }; const icon = iconMap[this.plugin.settings.iconStyle] || "\u{1F512}"; const lockIcon = this.lockView.createDiv({ cls: "peccirian-lock-icon" }); lockIcon.style.cssText = ` font-size: 64px; margin-bottom: 1.5rem; opacity: 0.8; transition: transform 0.3s ease; `; lockIcon.setText(icon); const message = this.lockView.createEl("p", { cls: "peccirian-message" }); message.style.cssText = ` margin-bottom: 2rem; font-size: 1.1rem; color: var(--text-muted); line-height: 1.5; `; message.setText(this.file ? `"${this.file.basename}" is an encrypted file. Click below to unlock.` : "This file is encrypted. Click below to unlock."); const unlockBtn = this.lockView.createEl("button", { cls: "peccirian-unlock-button" }); unlockBtn.style.cssText = ` padding: 0.75rem 1.5rem; font-size: 1rem; cursor: pointer; border-radius: 6px; border: 1px solid var(--background-modifier-border); background-color: var(--background-primary); color: var(--text-normal); transition: all 0.2s ease; box-shadow: 0 2px 4px rgba(0,0,0,0.1); `; unlockBtn.setText("Unlock"); unlockBtn.addEventListener("mouseenter", () => { unlockBtn.style.backgroundColor = "var(--background-secondary)"; unlockBtn.style.transform = "translateY(-1px)"; lockIcon.style.transform = "scale(1.1)"; }); unlockBtn.addEventListener("mouseleave", () => { unlockBtn.style.backgroundColor = "var(--background-primary)"; unlockBtn.style.transform = "translateY(0)"; lockIcon.style.transform = "scale(1)"; }); unlockBtn.addEventListener("click", async () => { await this.showPasswordModal(); }); this.isLockViewVisible = true; } hideLockView() { if (this.lockView) { this.lockView.style.display = "none"; this.isLockViewVisible = false; } const contentEl = this.contentEl; if (contentEl) { contentEl.style.pointerEvents = "auto"; } } async showPasswordModal() { if (this.isPasswordModalOpen || this.isLoadingFileInProgress) { return; } this.isLoadingFileInProgress = true; try { if (!this.file) { this.leaf.detach(); return; } const fileContents = await this.app.vault.read(this.file); try { this.encryptedData = JsonFileEncoding.decode(fileContents); if (!this.encryptedData || !this.encryptedData.encodedData) { this.leaf.detach(); return; } } catch (error) { this.leaf.detach(); return; } return new Promise((resolve, reject) => { var _a; if (this.isPasswordModalOpen) { reject(new Error("Password modal already open")); return; } this.isPasswordModalOpen = true; const hint = (_a = this.encryptedData) == null ? void 0 : _a.hint; const modal = new PasswordModal( this.app, async (password) => { try { if (!this.encryptedData) { this.leaf.detach(); reject(new Error("No encrypted data")); return; } const kdfOpts = this.plugin.settings.kdfType === "Argon2id" ? { type: "Argon2id", argon2MemoryKB: this.plugin.settings.argon2MemoryKB, argon2Iterations: this.plugin.settings.argon2Iterations, argon2Parallelism: this.plugin.settings.argon2Parallelism, argon2HashLen: this.plugin.settings.argon2HashLen } : { type: "PBKDF2", pbkdf2Iterations: this.plugin.settings.pbkdf2Iterations }; const decryptedText = await FileDataHelper.decrypt(this.encryptedData, password, this.plugin.getKeyCache() || void 0, kdfOpts); if (!decryptedText) { if (this.plugin.settings.showNotice) new import_obsidian8.Notice("Decryption failed: Password may be incorrect"); this.leaf.detach(); reject(new Error("Decryption failed")); return; } const processedText = decryptedText.trim() === "$\xB7-\xB7$" ? "" : decryptedText; this.setPassword(password); this.setUnencryptedViewData(processedText, true); this.isSavingEnabled = true; this.registerEditorChangeHandler(); this.hideLockView(); modal.close(); resolve(); } catch (err) { if (this.plugin.settings.showNotice) new import_obsidian8.Notice("Decryption failed: Unexpected error"); this.leaf.detach(); reject(err); } finally { this.isPasswordModalOpen = false; this.isLoadingFileInProgress = false; } }, () => { this.isPasswordModalOpen = false; this.isLoadingFileInProgress = false; this.leaf.detach(); resolve(); }, "permanent", true, false, false, this.plugin.settings.encryptionMethod, this.plugin, false, // Hide mode selection for permanent files hint // Pass the hint to display ); modal.open(); }); } catch (error) { this.isPasswordModalOpen = false; this.isLoadingFileInProgress = false; if (!(error instanceof Error && error.name === "UserCancelled")) { this.leaf.detach(); } } } async setState(state, result) { if (this.isLoadingFileInProgress || this.isPasswordModalOpen || this.isSwitchingView) { return; } const isViewSwitch = !this.isViewActive && ((state == null ? void 0 : state.file) instanceof import_obsidian8.TFile || typeof (state == null ? void 0 : state.file) === "string"); this.isViewActive = true; this.isSwitchingView = true; try { await super.setState(state, result); if (state == null ? void 0 : state.file) { if (state.file instanceof import_obsidian8.TFile) { this.file = state.file; this.filePath = state.file.path; this.updateTitle(state.file.basename); } else if (typeof state.file === "string") { this.filePath = state.file; const file = this.app.vault.getAbstractFileByPath(this.filePath); if (file instanceof import_obsidian8.TFile) { this.file = file; this.updateTitle(file.basename); } } } if (!this.isInitialized) { this.isInitialized = true; await this.onOpen(); } } finally { this.isSwitchingView = false; } } getState() { return { file: this.filePath, type: PECCIDIAN_VIEW_TYPE }; } async onLoadFile(file) { if (this.isSavingEnabled && this.password && this.file && this.file.path !== file.path && this.file.extension === "peccirian") { try { await this.save(false); } catch (error) { console.error("Failed to save before loading new file:", error); } } if (file.extension !== "peccirian") { this.handleNonPeccirianFile(file); return; } const isSameFile = this.lastOpenedFilePath === file.path; if (isSameFile) { return; } if (this.changeDebounceTimeout) { clearTimeout(this.changeDebounceTimeout); this.changeDebounceTimeout = null; } this.password = null; this.encryptedData = null; this.isSavingEnabled = false; this.isInitialized = false; this.lastOpenedFilePath = file.path; this.createLockView(); } async handleNonPeccirianFile(file) { if (this.isSavingEnabled && this.password && this.file && this.file.extension === "peccirian") { try { await this.save(false); } catch (error) { console.error("Failed to save before switching files:", error); } } if (this.changeDebounceTimeout) { clearTimeout(this.changeDebounceTimeout); this.changeDebounceTimeout = null; } this.isSavingEnabled = false; this.password = null; this.encryptedData = null; const currentLeaf = this.leaf; const newLeaf = this.app.workspace.getLeaf("tab"); await newLeaf.openFile(file); currentLeaf.detach(); } registerEditorChangeHandler() { setTimeout(() => { if (!this.editor) return; this.registerEvent( this.app.workspace.on("editor-change", (editor) => { if (editor === this.editor) { this.handleEditorChange(); } }) ); const cmEditor = this.editor.cm; if (cmEditor) { this.registerDomEvent(cmEditor.dom, "change", () => { this.handleEditorChange(); }); this.registerDomEvent(cmEditor.dom, "keyup", () => { this.handleEditorChange(); }); } }, 100); } handleEditorChange() { if (this.changeDebounceTimeout) { clearTimeout(this.changeDebounceTimeout); } this.changeDebounceTimeout = setTimeout(async () => { var _a, _b; if (!this.isSavingEnabled || !this.password) { return; } try { const currentContent = this.getUnencryptedViewData(); const password = this.getPassword(); if (!password) { throw new Error("Failed to retrieve password from memory"); } const kdfOpts2 = this.plugin.settings.kdfType === "Argon2id" ? { type: "Argon2id", argon2MemoryKB: this.plugin.settings.argon2MemoryKB, argon2Iterations: this.plugin.settings.argon2Iterations, argon2Parallelism: this.plugin.settings.argon2Parallelism, argon2HashLen: this.plugin.settings.argon2HashLen } : { type: "PBKDF2", pbkdf2Iterations: this.plugin.settings.pbkdf2Iterations }; this.encryptedData = await FileDataHelper.encrypt( password, currentContent, "permanent", ((_a = this.encryptedData) == null ? void 0 : _a.originalExt) || "md", (_b = this.encryptedData) == null ? void 0 : _b.hint, this.plugin.settings.encryptionMethod, // Pass algorithm from settings this.plugin.getKeyCache() || void 0, kdfOpts2 ); this.isSavingInProgress = true; try { const encryptedContent = JsonFileEncoding.encode(this.encryptedData); if (this.file) { await this.app.vault.modify(this.file, encryptedContent); } } finally { this.isSavingInProgress = false; } } catch (error) { console.error("Auto-encryption failed:", error); } }, 2e3); } outputStats(content) { const wordCount = content.split(/\s+/).filter((word) => word.length > 0).length; const charCount = content.length; const lineCount = content.split("\n").length; } async save(clear) { var _a, _b; if (this.isSavingInProgress) { return; } if (!this.file || !this.isSavingEnabled || !this.password) { return; } this.isSavingInProgress = true; try { let unencryptedDataToSave = this.getUnencryptedViewData(); const password = this.getPassword(); if (!password) { throw new Error("Failed to retrieve password from memory"); } if (this.plugin.settings.encryptAttachments) { const AttachmentHelper2 = (await Promise.resolve().then(() => (init_attachmentHelper(), attachmentHelper_exports))).AttachmentHelper; unencryptedDataToSave = AttachmentHelper2.updateAttachmentLinks( unencryptedDataToSave, "encrypt", "permanent" ); } const processedData = unencryptedDataToSave === "" ? "$\xB7-\xB7$" : unencryptedDataToSave; const kdfOpts3 = this.plugin.settings.kdfType === "Argon2id" ? { type: "Argon2id", argon2MemoryKB: this.plugin.settings.argon2MemoryKB, argon2Iterations: this.plugin.settings.argon2Iterations, argon2Parallelism: this.plugin.settings.argon2Parallelism, argon2HashLen: this.plugin.settings.argon2HashLen } : { type: "PBKDF2", pbkdf2Iterations: this.plugin.settings.pbkdf2Iterations }; this.encryptedData = await FileDataHelper.encrypt( password, processedData, "permanent", ((_a = this.file) == null ? void 0 : _a.extension) || "md", (_b = this.encryptedData) == null ? void 0 : _b.hint, this.plugin.settings.encryptionMethod, // Pass algorithm from settings this.plugin.getKeyCache() || void 0, kdfOpts3 ); if (!this.file) { return; } const encryptedContent = JsonFileEncoding.encode(this.encryptedData); await this.app.vault.modify(this.file, encryptedContent); } catch (error) { console.error("Save operation failed:", error); } finally { this.isSavingInProgress = false; } } getViewData() { if (this.isSavingInProgress && this.encryptedData) { return JsonFileEncoding.encode(this.encryptedData); } return this.getUnencryptedViewData(); } setViewData(data, clear) { if (JsonFileEncoding.isEncoded(data)) { try { const newEncoded = JsonFileEncoding.decode(data); if (!this.password) { this.leaf.detach(); return; } FileDataHelper.decrypt(newEncoded, this.password).then((decryptedText) => { if (decryptedText === null) { this.leaf.detach(); return; } this.setUnencryptedViewData(decryptedText, clear); }).catch((error) => { }); } catch (error) { } return; } this.setUnencryptedViewData(data, clear); } setUnencryptedViewData(data, clear) { super.setViewData(data, clear); if (this.file) { this.updateTitle(this.file.basename); } } getUnencryptedViewData() { return super.getViewData(); } detachSafely() { this.save(); this.isSavingEnabled = false; this.leaf.detach(); } async onOpen() { if (!this.file || !this.isViewActive || this.isSwitchingView) { return; } if (this.isPasswordModalOpen || this.isLoadingFileInProgress) { return; } try { await this.onLoadFile(this.file); } catch (error) { if (!(error instanceof Error && error.name === "UserCancelled")) { this.leaf.detach(); } } } async onunload() { var _a, _b; await super.onunload(); this.isViewActive = false; this.isSwitchingView = false; if (this.changeDebounceTimeout) { clearTimeout(this.changeDebounceTimeout); this.changeDebounceTimeout = null; } try { if (this.isSavingEnabled && this.password && this.file) { const currentContent = this.getUnencryptedViewData(); const password = this.getPassword(); if (!password) { return; } const kdfOpts4 = this.plugin.settings.kdfType === "Argon2id" ? { type: "Argon2id", argon2MemoryKB: this.plugin.settings.argon2MemoryKB, argon2Iterations: this.plugin.settings.argon2Iterations, argon2Parallelism: this.plugin.settings.argon2Parallelism, argon2HashLen: this.plugin.settings.argon2HashLen } : { type: "PBKDF2", pbkdf2Iterations: this.plugin.settings.pbkdf2Iterations }; this.encryptedData = await FileDataHelper.encrypt( password, currentContent, "permanent", ((_a = this.encryptedData) == null ? void 0 : _a.originalExt) || "md", (_b = this.encryptedData) == null ? void 0 : _b.hint, this.plugin.settings.encryptionMethod, // Pass algorithm from settings void 0, kdfOpts4 ); if (!this.file) { return; } const encryptedContent = JsonFileEncoding.encode(this.encryptedData); await this.app.vault.modify(this.file, encryptedContent); } } catch (error) { console.error("Error during final save:", error); } finally { this.memoryKey = null; this.password = null; this.encryptedData = null; this.isSavingEnabled = false; this.lastOpenedFilePath = null; } } setSavingEnabled(enabled) { this.isSavingEnabled = enabled; } isDecrypted() { return this.password !== null; } isDecrypting() { return this.isLoadingFileInProgress || this.isPasswordModalOpen; } async getDecryptedContent() { if (!this.isDecrypted()) { throw new Error("File is not decrypted"); } const content = this.getUnencryptedViewData(); return content; } getDecryptedPassword() { return this.getPassword(); } async onUnloadFile(file) { if (this.isInitialized && this.password && this.isSavingEnabled) { try { await super.onUnloadFile(file); } catch (error) { console.debug("Error during onUnloadFile:", error); } } } async detach() { await this.leaf.detach(); } async handlePasswordInput(password) { if (!this.file) return; try { let decryptedContent = await this.decryptFile(this.file, password); if (this.plugin.settings.encryptAttachments) { const AttachmentHelper2 = (await Promise.resolve().then(() => (init_attachmentHelper(), attachmentHelper_exports))).AttachmentHelper; decryptedContent = AttachmentHelper2.updateAttachmentLinks( decryptedContent, "decrypt", "permanent" ); } this.setViewData(decryptedContent, false); this.hideLockView(); this.setPassword(password); if (this.plugin.settings.showNotice) { new import_obsidian8.Notice("File decrypted"); } } catch (error) { console.error("Decryption failed:", error); if (this.plugin.settings.showNotice) { new import_obsidian8.Notice("Decryption failed: Password may be incorrect"); } } } async decryptFile(file, password) { try { const content = await this.app.vault.read(file); const data = JsonFileEncoding.decode(content); if (!data || !data.encodedData) { throw new Error("Invalid file format"); } const decryptedText = await FileDataHelper.decrypt(data, password); if (!decryptedText) { throw new Error("Decryption failed"); } return decryptedText; } catch (error) { console.error("Decryption error:", error); if (this.plugin.settings.showNotice) new import_obsidian8.Notice("Decryption failed: Unexpected error"); throw error; } } }; // src/main.ts init_fileData(); init_attachmentHelper(); init_keyCache(); var EccEncryptPlugin = class extends import_obsidian9.Plugin { constructor() { super(...arguments); this.toggleExtensionButton = null; this.keyCache = null; this.isConvertingToMarkdown = false; } // Mutex to prevent concurrent conversions async onload() { await this.loadSettings(); this.i18n = getI18n(window.localStorage.getItem("language") || "en"); if (this.settings.enableKeyCache) { this.keyCache = new KeyCache( this.settings.keyCacheTTL, this.settings.keyCacheMaxSize ); this.registerInterval(window.setInterval(() => { var _a; (_a = this.keyCache) == null ? void 0 : _a.cleanup(); }, 60 * 1e3)); } this.addCommand({ id: "encrypt-decrypt", name: "Toggle between .md and .eccirian", callback: async () => { const activeFile = this.app.workspace.getActiveFile(); if (!activeFile) { if (this.settings.showNotice) { new import_obsidian9.Notice("Please open a file first"); } return; } try { if (activeFile.extension === "eccirian") { const mdFile = await changeFileExtension(this.app.vault, activeFile, "md"); const leaf = this.app.workspace.getLeaf(); await leaf.openFile(mdFile); } else if (activeFile.extension === "md") { const eccFile = await changeFileExtension(this.app.vault, activeFile, "eccirian"); const leaf = this.app.workspace.getLeaf(); await leaf.openFile(eccFile); } else { if (this.settings.showNotice) new import_obsidian9.Notice("Only .md files can be converted to .eccirian"); } } catch (err) { } } }); this.addRibbonIcon(this.settings.iconStyle, "Encrypt/Decrypt Current File", async () => { await this.handleEncryptDecrypt(false); }); this.updateToggleExtensionButton(); this.registerView( ECCIDIAN_VIEW_TYPE, (leaf) => new EccidianView(leaf, this) ); this.registerView( PECCIDIAN_VIEW_TYPE, (leaf) => new PEccidianView(leaf, this) ); this.registerExtensions(["eccirian"], ECCIDIAN_VIEW_TYPE); this.registerExtensions(["peccirian"], PECCIDIAN_VIEW_TYPE); this.registerEvent( this.app.workspace.on("file-open", async (file) => { try { if (!file || !(file instanceof import_obsidian9.TFile)) return; if (file.extension !== "eccirian" && file.extension !== "peccirian") { return; } let leaf = this.app.workspace.getMostRecentLeaf(); if (!leaf) { leaf = this.app.workspace.getLeaf(); } const viewType = file.extension === "eccirian" ? ECCIDIAN_VIEW_TYPE : PECCIDIAN_VIEW_TYPE; await leaf.setViewState({ type: viewType, active: true }); const view = leaf.view; if (file.extension === "eccirian" && view instanceof EccidianView || file.extension === "peccirian" && view instanceof PEccidianView) { await view.setState({ file }, { history: true }); } else { console.error("View is not of correct type:", { extension: file.extension, viewType: view.getViewType() }); } } catch (error) { console.error("Error opening file:", error); if (this.settings.showNotice) { } } }) ); this.addSettingTab(new EccEncryptSettingTab(this.app, this)); this.registerDomEvent(document, "DOMContentLoaded", () => { this.loadStyles(); }); this.addCommand({ id: "encrypt-decrypt-file", name: "Encrypt/Decrypt file", callback: async () => { await this.handleEncryptDecrypt(true); } }); this.addCommand({ id: "encrypt-folder", name: "Encrypt Folder", checkCallback: (checking) => { const folder = this.getSelectedFolder(); if (folder) { if (!checking) { this.encryptFolder(folder); } return true; } return false; } }); this.addCommand({ id: "convert-to-md", name: "Remove Permanent Encryption (Convert to Normal Markdown)", checkCallback: (checking) => { const view = this.app.workspace.getActiveViewOfType(PEccidianView); if (view) { if (!checking) { this.convertToMarkdown(view); } return true; } return false; } }); this.addRibbonIcon("file-input", "Remove Permanent Encryption", async () => { const activeFile = this.app.workspace.getActiveFile(); if (!activeFile) { if (this.settings.showNotice) new import_obsidian9.Notice("Please open a file first"); return; } const activeView = this.app.workspace.getActiveViewOfType(PEccidianView); if (!activeView || activeView.file !== activeFile) { if (this.settings.showNotice) new import_obsidian9.Notice("Please open a decrypted permanently encrypted file first"); return; } try { this.convertToMarkdown(activeView); } catch (error) { console.error("Error converting file:", error); if (this.settings.showNotice) new import_obsidian9.Notice("Failed to convert file: " + (error instanceof Error ? error.message : "Unknown error")); } }); this.addCommand({ id: "convert-to-markdown", name: "Convert to Markdown", callback: async () => { const activeFile = this.app.workspace.getActiveFile(); if (!activeFile) { if (this.settings.showNotice) { new import_obsidian9.Notice("Please open a file first"); } return; } const activeView = this.app.workspace.getActiveViewOfType(PEccidianView); if (!activeView || activeView.file !== activeFile) { if (this.settings.showNotice) { new import_obsidian9.Notice("Please open a decrypted permanently encrypted file first"); } return; } try { await this.convertToMarkdown(activeView); } catch (error) { console.error("Error converting file:", error); if (this.settings.showNotice) { new import_obsidian9.Notice("Failed to convert file: " + (error instanceof Error ? error.message : "Unknown error")); } } } }); this.registerEvent( this.app.workspace.on("file-menu", (menu, file) => { if (file instanceof import_obsidian9.TFolder) { menu.addItem((item) => { item.setTitle(this.i18n.commandEncryptFolder).setIcon("lock").onClick(async () => { await this.encryptFolder(file); }); }); } }) ); } updateToggleExtensionButton() { if (this.toggleExtensionButton) { this.toggleExtensionButton.remove(); this.toggleExtensionButton = null; } if (this.settings.showToggleExtensionButton) { this.toggleExtensionButton = this.addRibbonIcon("file-text", "Toggle File Extension", async () => { const activeFile = this.app.workspace.getActiveFile(); if (!activeFile) { return; } try { if (activeFile.extension === "eccirian") { const mdFile = await changeFileExtension(this.app.vault, activeFile, "md"); const leaf = this.app.workspace.getLeaf(); await leaf.openFile(mdFile); } else if (activeFile.extension === "md") { const eccFile = await changeFileExtension(this.app.vault, activeFile, "eccirian"); const leaf = this.app.workspace.getLeaf(); await leaf.openFile(eccFile); } else { if (this.settings.showNotice) new import_obsidian9.Notice("Only .md files can be converted to .eccirian"); } } catch (err) { } }); } } async onunload() { if (this.keyCache) { this.keyCache.clear(); } if (this.toggleExtensionButton) { this.toggleExtensionButton.remove(); this.toggleExtensionButton = null; } this.app.workspace.detachLeavesOfType(ECCIDIAN_VIEW_TYPE); this.app.workspace.detachLeavesOfType(PECCIDIAN_VIEW_TYPE); } getKeyCache() { return this.keyCache; } async switchToMarkdownView(file) { this.app.workspace.detachLeavesOfType(ECCIDIAN_VIEW_TYPE); const leaf = this.app.workspace.getLeaf(); await leaf.openFile(file, { state: { mode: "source" } }); } async loadSettings() { this.settings = Object.assign({}, DEFAULT_SETTINGS, await this.loadData()); } async saveSettings() { await this.saveData(this.settings); } loadStyles() { const link = document.createElement("link"); link.rel = "stylesheet"; link.type = "text/css"; link.href = this.app.vault.adapter.getResourcePath(`${this.manifest.dir}/styles.css`); document.head.appendChild(link); } async convertToMarkdown(view) { var _a, _b, _c, _d, _e; if (this.isConvertingToMarkdown) { if (this.settings.showNotice) { new import_obsidian9.Notice("A conversion is already in progress. Please wait."); } return; } if (!view.isDecrypted() || view.isDecrypting()) { if (this.settings.showNotice) { new import_obsidian9.Notice("Please wait until decryption is complete."); } return; } this.isConvertingToMarkdown = true; try { if (!view.isDecrypted()) { if (this.settings.showNotice) { new import_obsidian9.Notice("Please decrypt the file first"); } return; } let content = await view.getDecryptedContent(); const currentFile = view.file; if (!currentFile) { if (this.settings.showNotice) { new import_obsidian9.Notice("Failed to get current file"); } return; } try { const raw = await this.app.vault.read(currentFile); if (raw.trim().startsWith("{") && raw.includes("encodedData")) { const dataObj = JsonFileEncoding.decode(raw); const originalExt = (dataObj.originalExt || "md").toLowerCase(); if (originalExt !== "md") { const password2 = view.getDecryptedPassword(); if (!password2) { if (this.settings.showNotice) new import_obsidian9.Notice("Missing password to restore file"); return; } const lines = String(dataObj.encodedData || "").split("\n"); const salt = ((_a = lines.find((l) => l.startsWith("SALT:"))) == null ? void 0 : _a.slice(5)) || ""; const iv = ((_b = lines.find((l) => l.startsWith("IV:"))) == null ? void 0 : _b.slice(3)) || ""; const data = ((_c = lines.find((l) => l.startsWith("DATA:"))) == null ? void 0 : _c.slice(5)) || ""; const fmt = (((_d = lines.find((l) => l.startsWith("FORMAT:"))) == null ? void 0 : _d.slice(7)) || "").toUpperCase(); const kdfLine = (_e = lines.find((l) => l.startsWith("KDF:"))) == null ? void 0 : _e.slice(4); if (!salt || !iv || !data) { if (this.settings.showNotice) new import_obsidian9.Notice("Invalid encrypted container"); return; } let finalKdfOpts = void 0; if (kdfLine) { const segs = kdfLine.split(":"); if (segs[0].toUpperCase() === "ARGON2ID" && segs.length >= 5) { finalKdfOpts = { type: "Argon2id", argon2MemoryKB: parseInt(segs[1], 10) || 65536, argon2Iterations: parseInt(segs[2], 10) || 3, argon2Parallelism: parseInt(segs[3], 10) || 1, argon2HashLen: parseInt(segs[4], 10) || 32 }; } else if (segs[0].toUpperCase() === "PBKDF2") { finalKdfOpts = { type: "PBKDF2", pbkdf2Iterations: parseInt(segs[1], 10) || 6e5 }; } } else if (dataObj.kdf) { if (dataObj.kdf === "Argon2id") { finalKdfOpts = { type: "Argon2id", argon2MemoryKB: dataObj.argon2MemoryKB || 65536, argon2Iterations: dataObj.argon2Iterations || 3, argon2Parallelism: dataObj.argon2Parallelism || 1, argon2HashLen: dataObj.argon2HashLen || 32 }; } else { finalKdfOpts = { type: "PBKDF2", pbkdf2Iterations: dataObj.iterations || 6e5 }; } } let arrayBuffer; if (fmt === "BIN") { const { decryptBytesWithPassword: decryptBytesWithPassword2 } = await Promise.resolve().then(() => (init_aesWithPassword(), aesWithPassword_exports)); const bytes = await decryptBytesWithPassword2(password2, salt, iv, data, this.keyCache || void 0, finalKdfOpts); arrayBuffer = bytes.buffer.slice(bytes.byteOffset, bytes.byteOffset + bytes.byteLength); } else { const base64 = await FileDataHelper.decrypt(dataObj, password2, this.keyCache || void 0, finalKdfOpts); if (!base64) { if (this.settings.showNotice) new import_obsidian9.Notice("Decryption failed: Password may be incorrect"); return; } const buf = Buffer.from(base64, "base64"); arrayBuffer = buf.buffer.slice(buf.byteOffset, buf.byteOffset + buf.byteLength); } const basePath2 = currentFile.path.replace(/\.peccirian$/, "").replace(/\.eccirian$/, ""); const originalPath = basePath2.toLowerCase().endsWith(".md") ? basePath2 : `${basePath2}.md`; const existing = this.app.vault.getAbstractFileByPath(originalPath); if (existing instanceof import_obsidian9.TFile) { const shouldOverwrite = await new Promise((resolve) => { const modal = new import_obsidian9.Modal(this.app); modal.titleEl.setText(this.i18n.dialogFileAlreadyExists || "File Already Exists"); const desc = modal.contentEl.createDiv(); const filename = originalPath.split("/").pop() || originalPath; desc.createEl("p", { text: (this.i18n.dialogFileAlreadyExistsDesc1 || 'A file named "{filename}" already exists.').replace("{filename}", filename) }); desc.createEl("p", { text: this.i18n.dialogFileAlreadyExistsDesc2 || "Decrypting will overwrite the existing file.", cls: "mod-warning" }); desc.createEl("p", { text: this.i18n.dialogFileAlreadyExistsDesc3 || "Do you want to continue?" }); const buttonContainer = modal.contentEl.createDiv({ cls: "modal-button-container" }); buttonContainer.style.display = "flex"; buttonContainer.style.justifyContent = "flex-end"; buttonContainer.style.gap = "8px"; buttonContainer.style.marginTop = "16px"; const cancelBtn = buttonContainer.createEl("button", { text: this.i18n.buttonCancel || "Cancel" }); cancelBtn.addEventListener("click", () => { modal.close(); resolve(false); }); const overwriteBtn = buttonContainer.createEl("button", { text: this.i18n.buttonContinue || "Continue", cls: "mod-warning" }); overwriteBtn.addEventListener("click", () => { modal.close(); resolve(true); }); modal.open(); }); if (!shouldOverwrite) { return; } await this.app.vault.delete(existing); } if (existing instanceof import_obsidian9.TFile) { await this.app.vault.modifyBinary(existing, arrayBuffer); } else { await this.app.vault.createBinary(originalPath, arrayBuffer); } view.setSavingEnabled(false); await this.app.vault.delete(currentFile); const leaf2 = this.app.workspace.getLeaf(); const restored = this.app.vault.getAbstractFileByPath(originalPath); if (restored instanceof import_obsidian9.TFile) await leaf2.openFile(restored); if (this.settings.showNotice) new import_obsidian9.Notice("Permanent encryption removed. Restored original file."); return; } } } catch (e) { } const password = view.getDecryptedPassword(); if (this.settings.encryptAttachments && password) { const contentWithEncryptedLinks = AttachmentHelper.updateAttachmentLinks( content, "encrypt", "permanent" ); const attachmentLinks = AttachmentHelper.extractAttachmentLinks(contentWithEncryptedLinks); await Promise.all( attachmentLinks.map(async (link) => { let encryptedAttachment = AttachmentHelper.resolveAttachmentPath( this.app.vault, currentFile.path, link ); if (encryptedAttachment) { try { await AttachmentHelper.decryptBinaryFile( this.app.vault, encryptedAttachment, password ); } catch (error) { console.error(`Failed to decrypt attachment ${link}:`, error); } } }) ); } content = AttachmentHelper.updateAttachmentLinks( content, "decrypt", "permanent" ); const basePath = currentFile.path.replace(/\.(peccirian|eccirian)$/i, ""); const mdPath = basePath.toLowerCase().endsWith(".md") ? basePath : `${basePath}.md`; const existingMdFile = this.app.vault.getAbstractFileByPath(mdPath); if (existingMdFile && existingMdFile instanceof import_obsidian9.TFile) { const shouldOverwrite = await new Promise((resolve) => { const modal = new import_obsidian9.Modal(this.app); modal.titleEl.setText(this.i18n.dialogFileAlreadyExists || "File Already Exists"); const desc = modal.contentEl.createDiv(); const filename = existingMdFile.basename; desc.createEl("p", { text: (this.i18n.dialogFileAlreadyExistsDesc1 || 'A file named "{filename}" already exists.').replace("{filename}", filename) }); desc.createEl("p", { text: this.i18n.dialogFileAlreadyExistsDesc2 || "Decrypting will overwrite the existing file.", cls: "mod-warning" }); desc.createEl("p", { text: this.i18n.dialogFileAlreadyExistsDesc3 || "Do you want to continue?" }); const buttonContainer = modal.contentEl.createDiv({ cls: "modal-button-container" }); buttonContainer.style.display = "flex"; buttonContainer.style.justifyContent = "flex-end"; buttonContainer.style.gap = "8px"; buttonContainer.style.marginTop = "16px"; const cancelBtn = buttonContainer.createEl("button", { text: this.i18n.buttonCancel || "Cancel" }); cancelBtn.addEventListener("click", () => { modal.close(); resolve(false); }); const overwriteBtn = buttonContainer.createEl("button", { text: this.i18n.buttonContinue || "Continue", cls: "mod-warning" }); overwriteBtn.addEventListener("click", () => { modal.close(); resolve(true); }); modal.open(); }); if (!shouldOverwrite) { return; } await this.app.vault.delete(existingMdFile); } await this.app.vault.rename(currentFile, mdPath); const mdFileAbs = this.app.vault.getAbstractFileByPath(mdPath); if (!(mdFileAbs instanceof import_obsidian9.TFile)) { throw new Error(`Failed to locate restored file at ${mdPath}`); } await this.app.vault.modify(mdFileAbs, content); view.setSavingEnabled(false); await view.detach(); const leaf = this.app.workspace.getLeaf(); await leaf.openFile(mdFileAbs); if (this.settings.showNotice) { new import_obsidian9.Notice("Permanent encryption removed. File is now a normal markdown file."); } } catch (error) { console.error("Error converting file:", error); if (this.settings.showNotice) { new import_obsidian9.Notice("Conversion failed: " + (error instanceof Error ? error.message : String(error))); } } finally { this.isConvertingToMarkdown = false; } } /** * Get currently selected folder from file explorer */ getSelectedFolder() { const fileExplorer = this.app.workspace.getLeavesOfType("file-explorer")[0]; if (fileExplorer) { const explorerView = fileExplorer.view; if (explorerView) { const fileItems = explorerView.fileItems; if (fileItems) { for (const [path, item] of Object.entries(fileItems)) { if (item.selfEl && item.selfEl.hasClass("is-active")) { const file = this.app.vault.getAbstractFileByPath(path); if (file instanceof import_obsidian9.TFolder) { return file; } } } } } } return null; } /** * Handle encryption/decryption for the active file * Used by both ribbon icon and command palette */ async handleEncryptDecrypt(showHintNotice = false) { const activeFile = this.app.workspace.getActiveFile(); if (!activeFile) { if (this.settings.showNotice) new import_obsidian9.Notice("Please open a file first"); return; } const activeView = this.app.workspace.getActiveViewOfType(PEccidianView); if (activeView && activeView.file === activeFile && activeView.isDecrypted()) { if (this.settings.showNotice) new import_obsidian9.Notice("This file is already decrypted"); return; } if (activeFile.extension === "peccirian") { if (activeView) { const unlockBtn = activeView.contentEl.querySelector(".peccirian-unlock-button"); if (unlockBtn) { unlockBtn.click(); } return; } } const buildKdfOptsFromSettings = () => { if (this.settings.kdfType === "Argon2id") { return { type: "Argon2id", argon2MemoryKB: this.settings.argon2MemoryKB, argon2Iterations: this.settings.argon2Iterations, argon2Parallelism: this.settings.argon2Parallelism, argon2HashLen: this.settings.argon2HashLen }; } return { type: "PBKDF2", pbkdf2Iterations: this.settings.pbkdf2Iterations }; }; const kdfOpts = buildKdfOptsFromSettings(); if (activeFile.extension !== "md" && activeFile.extension !== "eccirian" && activeFile.extension !== "peccirian") { new PasswordModal( this.app, async (password, encryptionMethod, hint, encryptionMode) => { const selectedMode = encryptionMode || this.settings.defaultEncryptionMode; try { const bytes = await this.app.vault.readBinary(activeFile); const { encryptBytesWithPassword: encryptBytesWithPassword2 } = await Promise.resolve().then(() => (init_aesWithPassword(), aesWithPassword_exports)); const { salt, iv, data } = await encryptBytesWithPassword2(password, bytes, this.keyCache || void 0, kdfOpts); let kdfLine = ""; if (kdfOpts.type === "Argon2id") { const mem = kdfOpts.argon2MemoryKB || 65536; const iters = kdfOpts.argon2Iterations || 3; const para = kdfOpts.argon2Parallelism || 1; const len = kdfOpts.argon2HashLen || 32; kdfLine = `KDF:Argon2id:${mem}:${iters}:${para}:${len}`; } else { kdfLine = `KDF:PBKDF2:${kdfOpts.pbkdf2Iterations || 6e5}`; } const encodedData = `SALT:${salt} IV:${iv} DATA:${data} FORMAT:BIN ${kdfLine}`; const container = { version: FileDataHelper.DEFAULT_VERSION, encodedData, type: selectedMode, originalExt: activeFile.extension, algorithm: "AES", kdf: kdfOpts.type === "PBKDF2" ? "PBKDF2" : "Argon2id", iterations: kdfOpts.type === "PBKDF2" ? kdfOpts.pbkdf2Iterations || 6e5 : void 0, argon2MemoryKB: kdfOpts.type === "Argon2id" ? kdfOpts.argon2MemoryKB || 65536 : void 0, argon2Iterations: kdfOpts.type === "Argon2id" ? kdfOpts.argon2Iterations || 3 : void 0, argon2Parallelism: kdfOpts.type === "Argon2id" ? kdfOpts.argon2Parallelism || 1 : void 0, argon2HashLen: kdfOpts.type === "Argon2id" ? kdfOpts.argon2HashLen || 32 : void 0 }; const ext = selectedMode === "permanent" ? "peccirian" : "eccirian"; const encPath = `${activeFile.path}.${ext}`; await this.app.vault.create(encPath, JSON.stringify(container, null, 2)); await this.app.vault.delete(activeFile); if (this.settings.showNotice) new import_obsidian9.Notice("File encrypted"); const encFile = this.app.vault.getAbstractFileByPath(encPath); if (encFile instanceof import_obsidian9.TFile) { const leaf = this.app.workspace.getLeaf(); await leaf.openFile(encFile); } } catch (error) { console.error("Binary encryption failed", error); if (this.settings.showNotice) new import_obsidian9.Notice("Encryption failed"); } }, () => { }, this.settings.defaultEncryptionMode, false, this.settings.requirePasswordConfirmation, this.settings.showHint, this.settings.encryptionMethod, this, true, void 0 ).open(); return; } let fileContent; const mdView = this.app.workspace.getActiveViewOfType(import_obsidian9.MarkdownView); if (mdView && mdView.file === activeFile && mdView.editor) { fileContent = mdView.editor.getValue(); } else { fileContent = await this.app.vault.read(activeFile); } const isEncryptedWithENC = fileContent.includes("%%ENC"); const isEncryptedWithJSON = fileContent.trim().startsWith("{") && fileContent.includes("encodedData"); if (isEncryptedWithENC || isEncryptedWithJSON) { const hintMatch = fileContent.match(/HINT:(.+)/); const hint = hintMatch ? hintMatch[1] : void 0; if (hint && showHintNotice && this.settings.showNotice) { new import_obsidian9.Notice(`Password Hint: ${hint}`); } new PasswordModal( this.app, async (password, encryptionMethod, hintInput, encryptionMode) => { try { if (isEncryptedWithENC) { const saltMatch = fileContent.match(/SALT:(.+)/); const ivMatch = fileContent.match(/IV:(.+)/); const dataMatch = fileContent.match(/DATA:(.+)/); const publicKeyMatch = fileContent.match(/PUBLIC_KEY:(.+)/); const ephemeralPublicKeyMatch = fileContent.match(/EPHEMERAL_PUBLIC_KEY:(.+)/); if (saltMatch && ivMatch && dataMatch) { const typeMatch = fileContent.match(/TYPE:(.+)/); const encryptionType = typeMatch ? typeMatch[1] : "temporary"; const originalExtMatch = fileContent.match(/ORIGINAL_EXT:(.+)/); const hintMatchInEnc = fileContent.match(/HINT:(.+)/); const publicKey = publicKeyMatch ? publicKeyMatch[1] : void 0; const ephemeral = ephemeralPublicKeyMatch ? ephemeralPublicKeyMatch[1] : void 0; const encodedLines = [ `SALT:${saltMatch[1]}`, `IV:${ivMatch[1]}`, `DATA:${dataMatch[1]}`, ...publicKey ? [`PUBLIC_KEY:${publicKey}`] : [], ...ephemeral ? [`EPHEMERAL_PUBLIC_KEY:${ephemeral}`] : [] ]; const fileData = { version: FileDataHelper.DEFAULT_VERSION, encodedData: encodedLines.join("\n"), type: encryptionType, originalExt: originalExtMatch ? originalExtMatch[1] : "md", hint: hintMatchInEnc ? hintMatchInEnc[1] : void 0, algorithm: publicKey ? "ECC" : "AES" }; const decrypted = await FileDataHelper.decrypt(fileData, password, this.keyCache || void 0, kdfOpts); if (!decrypted) { if (this.settings.showNotice) { new import_obsidian9.Notice("Decryption failed: Password may be incorrect"); } return; } let processedText = decrypted.trim() === "$\xB7-\xB7$" ? "" : decrypted; const encryptionTypeStr = encryptionType; if (this.settings.encryptAttachments) { const attachmentLinks = AttachmentHelper.extractAttachmentLinks(processedText); await Promise.all( attachmentLinks.map(async (link) => { let encryptedAttachment = AttachmentHelper.resolveAttachmentPath( this.app.vault, activeFile.path, link ); if (encryptedAttachment) { try { await AttachmentHelper.decryptBinaryFile( this.app.vault, encryptedAttachment, password ); } catch (error) { console.error(`Failed to decrypt attachment ${link}:`, error); } } else { console.warn(`Attachment not found: ${link}`); } }) ); processedText = AttachmentHelper.updateAttachmentLinks( processedText, "decrypt", encryptionTypeStr ); } const mdFile = await changeFileExtension(this.app.vault, activeFile, "md"); await this.app.vault.modify(mdFile, processedText); if (this.settings.showNotice) { new import_obsidian9.Notice("File decrypted"); } const leaf = this.app.workspace.getLeaf(); await leaf.openFile(mdFile); } else { if (this.settings.showNotice) { new import_obsidian9.Notice("Decryption failed: Invalid file format"); } } } else if (isEncryptedWithJSON) { try { const encryptedData = JSON.parse(fileContent); if (!encryptedData || !encryptedData.encodedData) { if (this.settings.showNotice) { new import_obsidian9.Notice("Decryption failed: Invalid JSON format"); } return; } const decryptedText = await FileDataHelper.decrypt(encryptedData, password, this.keyCache || void 0, kdfOpts); if (!decryptedText) { if (this.settings.showNotice) { new import_obsidian9.Notice("Decryption failed: Password may be incorrect"); } return; } const processedText = decryptedText.trim() === "$\xB7-\xB7$" ? "" : decryptedText; await this.app.vault.modify(activeFile, processedText); const mdFile = await changeFileExtension(this.app.vault, activeFile, "md"); if (this.settings.showNotice) { new import_obsidian9.Notice("File decrypted"); } const leaf = this.app.workspace.getLeaf(); await leaf.openFile(mdFile); } catch (error) { if (this.settings.showNotice) { new import_obsidian9.Notice("Decryption failed: " + (error instanceof Error ? error.message : "Unknown error")); } } } } catch (err) { if (err instanceof Error && err.message.includes("Error Password")) { if (this.settings.showNotice) { new import_obsidian9.Notice("Operation failed: Password may be incorrect"); } } } }, () => { }, this.settings.defaultEncryptionMode, true, this.settings.requirePasswordConfirmation, this.settings.showHint, this.settings.encryptionMethod, this, true, hint ).open(); } else { new PasswordModal( this.app, async (password, encryptionMethod, hint, encryptionMode) => { try { let contentToEncrypt = fileContent || "$\xB7-\xB7$"; const selectedMode = encryptionMode || this.settings.defaultEncryptionMode; if (this.settings.encryptAttachments) { const attachmentLinks = AttachmentHelper.extractAttachmentLinks(contentToEncrypt); await Promise.all( attachmentLinks.map(async (link) => { const attachmentFile = AttachmentHelper.resolveAttachmentPath( this.app.vault, activeFile.path, link ); if (attachmentFile) { try { await AttachmentHelper.encryptBinaryFile( this.app.vault, attachmentFile, password, selectedMode, kdfOpts ); } catch (error) { console.error(`Failed to encrypt attachment ${link}:`, error); } } }) ); contentToEncrypt = AttachmentHelper.updateAttachmentLinks( contentToEncrypt, "encrypt", selectedMode ); } if (encryptionMethod === "AES") { const { salt, iv, data } = await encryptWithPassword(password, contentToEncrypt, this.keyCache || void 0, kdfOpts); let kdfLine = ""; if (kdfOpts.type === "Argon2id") { const mem = kdfOpts.argon2MemoryKB || 65536; const iters = kdfOpts.argon2Iterations || 3; const para = kdfOpts.argon2Parallelism || 1; const len = kdfOpts.argon2HashLen || 32; kdfLine = ` KDF:Argon2id:${mem}:${iters}:${para}:${len}`; } else { kdfLine = ` KDF:PBKDF2:${kdfOpts.pbkdf2Iterations || 6e5}`; } const encrypted = `%%ENC TYPE:${selectedMode} ORIGINAL_EXT:${activeFile.extension} SALT:${salt} IV:${iv} DATA:${data}${kdfLine}${hint ? ` HINT:${hint}` : ""}`; const encFile = await changeFileExtension(this.app.vault, activeFile, this.settings.fileExtension); await this.app.vault.modify(encFile, encrypted); if (this.settings.showNotice) new import_obsidian9.Notice("File encrypted"); const leaf = this.app.workspace.getLeaf(); await leaf.openFile(encFile); } else { const { salt, iv, data, publicKey, ephemeralPublicKey } = await encryptWithPassword2(password, contentToEncrypt, this.keyCache || void 0, kdfOpts); let kdfLine = ""; if (kdfOpts.type === "Argon2id") { const mem = kdfOpts.argon2MemoryKB || 65536; const iters = kdfOpts.argon2Iterations || 3; const para = kdfOpts.argon2Parallelism || 1; const len = kdfOpts.argon2HashLen || 32; kdfLine = ` KDF:Argon2id:${mem}:${iters}:${para}:${len}`; } else { kdfLine = ` KDF:PBKDF2:${kdfOpts.pbkdf2Iterations || 6e5}`; } const encrypted = `%%ENC TYPE:${selectedMode} ORIGINAL_EXT:${activeFile.extension} SALT:${salt} IV:${iv} DATA:${data} PUBLIC_KEY:${publicKey} EPHEMERAL_PUBLIC_KEY:${ephemeralPublicKey}${kdfLine}${hint ? ` HINT:${hint}` : ""}`; const encFile = await changeFileExtension(this.app.vault, activeFile, this.settings.fileExtension); await this.app.vault.modify(encFile, encrypted); if (this.settings.showNotice) { new import_obsidian9.Notice("File encrypted"); } const leaf = this.app.workspace.getLeaf(); await leaf.openFile(encFile); } } catch (err) { if (err instanceof Error && err.message.includes("Error Password")) { if (this.settings.showNotice) { new import_obsidian9.Notice("Operation failed: Password may be incorrect"); } } } }, () => { }, this.settings.defaultEncryptionMode, false, this.settings.requirePasswordConfirmation, this.settings.showHint, this.settings.encryptionMethod, this, true, void 0 ).open(); } } /** * Encrypt a folder * Note: Folder encryption only supports temporary mode. * Permanent mode is not supported for folders due to memory and complexity constraints. */ async encryptFolder(folder) { const kdfOpts = this.settings.kdfType === "Argon2id" ? { type: "Argon2id", argon2MemoryKB: this.settings.argon2MemoryKB, argon2Iterations: this.settings.argon2Iterations, argon2Parallelism: this.settings.argon2Parallelism, argon2HashLen: this.settings.argon2HashLen } : { type: "PBKDF2", pbkdf2Iterations: this.settings.pbkdf2Iterations }; new PasswordModal( this.app, async (password, encryptionMethod, hint, encryptionMode) => { try { const folderContent = await FolderHelper.packFolder(this.app.vault, folder); const selectedMode = "temporary"; let kdfLine = ""; if (kdfOpts.type === "Argon2id") { const mem = kdfOpts.argon2MemoryKB || 65536; const iters = kdfOpts.argon2Iterations || 3; const para = kdfOpts.argon2Parallelism || 1; const len = kdfOpts.argon2HashLen || 32; kdfLine = ` KDF:Argon2id:${mem}:${iters}:${para}:${len}`; } else { kdfLine = ` KDF:PBKDF2:${kdfOpts.pbkdf2Iterations || 6e5}`; } if (encryptionMethod === "AES") { const { salt, iv, data } = await encryptWithPassword(password, folderContent, this.keyCache || void 0, kdfOpts); const encrypted = `%%ENC TYPE:${selectedMode} IS_FOLDER:true ORIGINAL_PATH:${folder.path} SALT:${salt} IV:${iv} DATA:${data}${kdfLine}${hint ? ` HINT:${hint}` : ""}`; let encryptedFileName = `${folder.path}.${this.settings.fileExtension}`; let n = 1; while (this.app.vault.getAbstractFileByPath(encryptedFileName)) { const basePath = `${folder.path}.${this.settings.fileExtension}`; encryptedFileName = `${basePath.replace(`.${this.settings.fileExtension}$`, "")} (${n}).${this.settings.fileExtension}`; n++; } await this.app.vault.create(encryptedFileName, encrypted); await this.app.vault.delete(folder, true); if (this.settings.showNotice) { new import_obsidian9.Notice(this.i18n.messageFolderEncrypted); } const encFile = this.app.vault.getAbstractFileByPath(encryptedFileName); if (encFile instanceof import_obsidian9.TFile) { const leaf = this.app.workspace.getLeaf(); await leaf.openFile(encFile); } } else { const { salt, iv, data, publicKey, ephemeralPublicKey } = await encryptWithPassword2(password, folderContent, this.keyCache || void 0, kdfOpts); const encrypted = `%%ENC TYPE:${selectedMode} IS_FOLDER:true ORIGINAL_PATH:${folder.path} SALT:${salt} IV:${iv} DATA:${data} PUBLIC_KEY:${publicKey} EPHEMERAL_PUBLIC_KEY:${ephemeralPublicKey}${kdfLine}${hint ? ` HINT:${hint}` : ""}`; let encryptedFileName = `${folder.path}.${this.settings.fileExtension}`; let n = 1; while (this.app.vault.getAbstractFileByPath(encryptedFileName)) { const basePath = `${folder.path}.${this.settings.fileExtension}`; encryptedFileName = `${basePath.replace(`.${this.settings.fileExtension}$`, "")} (${n}).${this.settings.fileExtension}`; n++; } await this.app.vault.create(encryptedFileName, encrypted); await this.app.vault.delete(folder, true); if (this.settings.showNotice) { new import_obsidian9.Notice(this.i18n.messageFolderEncrypted); } const encFile = this.app.vault.getAbstractFileByPath(encryptedFileName); if (encFile instanceof import_obsidian9.TFile) { const leaf = this.app.workspace.getLeaf(); await leaf.openFile(encFile, { active: true }); } } } catch (error) { console.error("Error encrypting folder:", error); if (this.settings.showNotice) new import_obsidian9.Notice("Folder encryption failed: " + (error instanceof Error ? error.message : "Unknown error")); } }, () => { }, "temporary", // Force temporary mode for folders false, this.settings.requirePasswordConfirmation, this.settings.showHint, this.settings.encryptionMethod, this, false, // Hide mode selection for folder encryption void 0 // No existing hint ).open(); } }; /* nosourcemap */