mirror of
https://github.com/vrtmrz/obsidian-livesync.git
synced 2026-06-17 11:50:16 +00:00
Compare commits
25 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 72033472f3 | |||
| 93dc03e86f | |||
| dae8443fe8 | |||
| 88a8bcbd5a | |||
| 4a5283543d | |||
| 7895336189 | |||
| 2d5cdccf7d | |||
| 497fd04081 | |||
| ae9c46f8f0 | |||
| dcd10cd690 | |||
| 5a35b71339 | |||
| 18a59219f7 | |||
| c9095738e6 | |||
| bed415fc6b | |||
| 4eeb93b9e4 | |||
| 0fc233686b | |||
| def1d297f5 | |||
| e7cf2a6fba | |||
| 2da2fd7671 | |||
| d5175969e7 | |||
| 46f9630999 | |||
| 75adcf0ff0 | |||
| 866dd49ba3 | |||
| d0a84d07aa | |||
| 089a4c0f8b |
@@ -12,6 +12,17 @@ on:
|
|||||||
- main
|
- main
|
||||||
tags:
|
tags:
|
||||||
- "*.*.*-cli"
|
- "*.*.*-cli"
|
||||||
|
paths-ignore:
|
||||||
|
- "docs/**"
|
||||||
|
- "*.md"
|
||||||
|
- "images/**"
|
||||||
|
- "assets/**"
|
||||||
|
- "instruction_images/**"
|
||||||
|
- "src/apps/webapp/**"
|
||||||
|
- "src/apps/webpeer/**"
|
||||||
|
- ".github/workflows/release.yml"
|
||||||
|
- ".github/workflows/unit-ci.yml"
|
||||||
|
- ".github/workflows/harness-ci.yml"
|
||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
inputs:
|
inputs:
|
||||||
dry_run:
|
dry_run:
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ on:
|
|||||||
# Sequence of patterns matched against refs/tags
|
# Sequence of patterns matched against refs/tags
|
||||||
tags:
|
tags:
|
||||||
- '*' # Push events to matching any tag format, i.e. 1.0, 20.15.10
|
- '*' # Push events to matching any tag format, i.e. 1.0, 20.15.10
|
||||||
|
- '!*-cli' # Exclude command-line interface tags
|
||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
|
|||||||
+35
-2
@@ -19,11 +19,15 @@ const packageJson = JSON.parse(fs.readFileSync("./package.json") + "");
|
|||||||
const updateInfo = JSON.stringify(fs.readFileSync("./updates.md") + "");
|
const updateInfo = JSON.stringify(fs.readFileSync("./updates.md") + "");
|
||||||
|
|
||||||
const PATHS_TEST_INSTALL = process.env?.PATHS_TEST_INSTALL || "";
|
const PATHS_TEST_INSTALL = process.env?.PATHS_TEST_INSTALL || "";
|
||||||
const PATH_TEST_INSTALL = PATHS_TEST_INSTALL.split(path.delimiter).map(p => p.trim()).filter(p => p.length);
|
const PATH_TEST_INSTALL = PATHS_TEST_INSTALL.split(path.delimiter)
|
||||||
|
.map((p) => p.trim())
|
||||||
|
.filter((p) => p.length);
|
||||||
if (PATH_TEST_INSTALL) {
|
if (PATH_TEST_INSTALL) {
|
||||||
console.log(`Built files will be copied to ${PATH_TEST_INSTALL}`);
|
console.log(`Built files will be copied to ${PATH_TEST_INSTALL}`);
|
||||||
} else {
|
} else {
|
||||||
console.log("Development build: You can install the plug-in to Obsidian for testing by exporting the PATHS_TEST_INSTALL environment variable with the paths to your vault plugins directories separated by your system path delimiter (':' on Unix, ';' on Windows).");
|
console.log(
|
||||||
|
"Development build: You can install the plug-in to Obsidian for testing by exporting the PATHS_TEST_INSTALL environment variable with the paths to your vault plugins directories separated by your system path delimiter (':' on Unix, ';' on Windows)."
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const moduleAliasPlugin = {
|
const moduleAliasPlugin = {
|
||||||
@@ -66,6 +70,34 @@ const moduleAliasPlugin = {
|
|||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const removePragmaCommentsPlugin = {
|
||||||
|
name: "remove-pragma-comments",
|
||||||
|
setup(build) {
|
||||||
|
// Filter target extensions (e.g., JavaScript and TypeScript)
|
||||||
|
build.onLoad({ filter: /\.[jt]s?$/ }, async (args) => {
|
||||||
|
const source = await fs.promises.readFile(args.path, "utf8");
|
||||||
|
|
||||||
|
// Regex targeting both single-line and multi-line comments
|
||||||
|
// This regex looks for:
|
||||||
|
// - /* eslint ... */ (multi-line)
|
||||||
|
// const esLintPragmaRegexBlock = /\/\*[\s\S]*?eslint[\s\S]*?\*\/|([^\\:]|^)\/\/.*eslint.*$/gm;
|
||||||
|
// - // eslint-disable-next-line
|
||||||
|
let cleanedSource = source;
|
||||||
|
const tsIgnoreRegex = /\/\*\s*@ts-ignore\s*\*\/|([^\\:]|^)\/\/.*?@ts-ignore.*$/gm;
|
||||||
|
const esLintPragmaRegexLine = /([^\\:]|^)\/\/.*?eslint-.*$/gm;
|
||||||
|
const exps = [tsIgnoreRegex, esLintPragmaRegexLine];
|
||||||
|
for (const exp of exps) {
|
||||||
|
cleanedSource = cleanedSource.replace(exp, "$1");
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
contents: cleanedSource,
|
||||||
|
loader: args.path.endsWith("ts") ? "ts" : "js",
|
||||||
|
};
|
||||||
|
});
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
/** @type esbuild.Plugin[] */
|
/** @type esbuild.Plugin[] */
|
||||||
const plugins = [
|
const plugins = [
|
||||||
{
|
{
|
||||||
@@ -177,6 +209,7 @@ const context = await esbuild.context({
|
|||||||
preprocess: sveltePreprocess(),
|
preprocess: sveltePreprocess(),
|
||||||
compilerOptions: { css: "injected", preserveComments: false },
|
compilerOptions: { css: "injected", preserveComments: false },
|
||||||
}),
|
}),
|
||||||
|
removePragmaCommentsPlugin,
|
||||||
...plugins,
|
...plugins,
|
||||||
],
|
],
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -0,0 +1,143 @@
|
|||||||
|
const restrictedGlobalsOptions = [
|
||||||
|
{
|
||||||
|
name: "app",
|
||||||
|
message: "Avoid using the global app object. Instead use the reference provided by your plugin instance.",
|
||||||
|
},
|
||||||
|
"warn",
|
||||||
|
{
|
||||||
|
name: "fetch",
|
||||||
|
message: "Use the built-in `requestUrl` function instead of `fetch` for network requests in Obsidian.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "localStorage",
|
||||||
|
message:
|
||||||
|
"Prefer `App#saveLocalStorage` / `App#loadLocalStorage` functions to write / read localStorage data that's unique to a vault.",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
const restrictedImportsOptions = [
|
||||||
|
{
|
||||||
|
name: "axios",
|
||||||
|
message: "Use the built-in `requestUrl` function instead of `axios`.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "superagent",
|
||||||
|
message: "Use the built-in `requestUrl` function instead of `superagent`.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "got",
|
||||||
|
message: "Use the built-in `requestUrl` function instead of `got`.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "ofetch",
|
||||||
|
message: "Use the built-in `requestUrl` function instead of `ofetch`.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "ky",
|
||||||
|
message: "Use the built-in `requestUrl` function instead of `ky`.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "node-fetch",
|
||||||
|
message: "Use the built-in `requestUrl` function instead of `node-fetch`.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "moment",
|
||||||
|
message: "The 'moment' package is bundled with Obsidian. Please import it from 'obsidian' instead.",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const warnWhileDev = "off";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @type {import("eslint").Linter.RulesRecord}
|
||||||
|
*/
|
||||||
|
export const baseRules = {
|
||||||
|
// -- Base rules (turned off in favour of TS specific versions or explicitly disabled).
|
||||||
|
"no-unused-vars": "off",
|
||||||
|
"no-unused-labels": "off",
|
||||||
|
"no-prototype-builtins": "off",
|
||||||
|
"require-await": "off",
|
||||||
|
// -- TypeScript specific rules (Gradual adoption of stricter rules, currently set to 'warn' for a while).
|
||||||
|
"@typescript-eslint/no-explicit-any": "warn",
|
||||||
|
"@typescript-eslint/no-redundant-type-constituents": "warn",
|
||||||
|
// -- TypeScript specific rules
|
||||||
|
// @typescript-eslint/no-unsafe-* rules and @typescript-eslint/no-explicit-any:
|
||||||
|
// This project contains a lot of library-sh code where the use of `any` is often necessary and justified.
|
||||||
|
// Rules is now set to 'off' for a while.
|
||||||
|
"@typescript-eslint/no-unsafe-argument": "off",
|
||||||
|
"@typescript-eslint/no-unsafe-call": "off",
|
||||||
|
"@typescript-eslint/no-unsafe-member-access": "off",
|
||||||
|
"@typescript-eslint/no-unsafe-return": "off",
|
||||||
|
"@typescript-eslint/no-unsafe-assignment": "off",
|
||||||
|
// -- Reasonable rules.
|
||||||
|
"@typescript-eslint/no-deprecated": warnWhileDev,
|
||||||
|
"@typescript-eslint/no-unused-vars": ["error", { args: "none" }],
|
||||||
|
"@typescript-eslint/ban-ts-comment": "off",
|
||||||
|
"@typescript-eslint/no-empty-function": "off",
|
||||||
|
"@typescript-eslint/require-await": "error",
|
||||||
|
"@typescript-eslint/no-misused-promises": "error",
|
||||||
|
"@typescript-eslint/no-floating-promises": "error",
|
||||||
|
"@typescript-eslint/no-unnecessary-type-assertion": "error",
|
||||||
|
|
||||||
|
// -- General rules
|
||||||
|
"no-async-promise-executor": warnWhileDev,
|
||||||
|
"no-constant-condition": ["error", { checkLoops: false }],
|
||||||
|
// -- Disabled rules
|
||||||
|
// no-undef: This option breaks the global declarations for the library files and is not worth the effort to fix at this time.
|
||||||
|
"no-undef": "off",
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @type {import("eslint").Linter.RulesRecord}
|
||||||
|
*/
|
||||||
|
export const obsidianRules = {
|
||||||
|
// -- Obsidian rules
|
||||||
|
// obsidianmd/no-unsupported-api: usually this project checks for API support at runtime, so this rule is not critical but can be helpful to catch potential issues.
|
||||||
|
"obsidianmd/no-unsupported-api": warnWhileDev,
|
||||||
|
|
||||||
|
// -- Plugin specific overrides
|
||||||
|
"obsidianmd/rule-custom-message": "off",
|
||||||
|
"obsidianmd/ui/sentence-case": "off",
|
||||||
|
"obsidianmd/no-plugin-as-component": "off",
|
||||||
|
|
||||||
|
// -- Temporary overrides for migration
|
||||||
|
"obsidianmd/no-static-styles-assignment": "off",
|
||||||
|
};
|
||||||
|
/**
|
||||||
|
* @type {(base:string) => import("eslint").Linter.RulesRecord}
|
||||||
|
*/
|
||||||
|
export const ImportAliasRules = (base) => ({
|
||||||
|
"@dword-design/import-alias/prefer-alias": [
|
||||||
|
"error",
|
||||||
|
{
|
||||||
|
aliasForSubpaths: true,
|
||||||
|
alias: {
|
||||||
|
"@": `${base}/src`,
|
||||||
|
"@lib": `${base}/src/lib/src`,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
/**
|
||||||
|
* @type {import("eslint").Linter.RulesRecord}
|
||||||
|
*/
|
||||||
|
export const CommunityReviewRecommendedRules = {
|
||||||
|
"no-unused-vars": "off",
|
||||||
|
"no-prototype-bultins": "off",
|
||||||
|
"no-self-compare": "warn",
|
||||||
|
"no-eval": "error",
|
||||||
|
"no-implied-eval": "error",
|
||||||
|
"prefer-const": "off",
|
||||||
|
"no-implicit-globals": "error",
|
||||||
|
"no-console": "off", // overridden by obsidianmd/rule-custom-message
|
||||||
|
"no-restricted-globals": ["error", ...restrictedGlobalsOptions],
|
||||||
|
"no-restricted-imports": ["error", ...restrictedImportsOptions],
|
||||||
|
"no-alert": "error",
|
||||||
|
"no-undef": "error",
|
||||||
|
"@typescript-eslint/ban-ts-comment": "off",
|
||||||
|
"@typescript-eslint/no-deprecated": "error",
|
||||||
|
"@typescript-eslint/no-unused-vars": ["warn", { args: "none" }],
|
||||||
|
"@typescript-eslint/require-await": "off",
|
||||||
|
"@typescript-eslint/no-explicit-any": ["error", { fixToUnknown: true }],
|
||||||
|
// "import/no-nodejs-modules": "off",
|
||||||
|
// "import/no-extraneous-dependencies": "error",
|
||||||
|
};
|
||||||
+17
-49
@@ -4,6 +4,8 @@ import globals from "globals";
|
|||||||
import { defineConfig, globalIgnores } from "eslint/config";
|
import { defineConfig, globalIgnores } from "eslint/config";
|
||||||
import * as sveltePlugin from "eslint-plugin-svelte";
|
import * as sveltePlugin from "eslint-plugin-svelte";
|
||||||
import svelteParser from "svelte-eslint-parser";
|
import svelteParser from "svelte-eslint-parser";
|
||||||
|
import importAlias from "@dword-design/eslint-plugin-import-alias";
|
||||||
|
import { baseRules, ImportAliasRules, obsidianRules } from "./eslint.config.common.mjs";
|
||||||
const warnWhileDev = "off"; // Change to "warn" to enable warnings for rules that are currently disabled.
|
const warnWhileDev = "off"; // Change to "warn" to enable warnings for rules that are currently disabled.
|
||||||
export default defineConfig([
|
export default defineConfig([
|
||||||
globalIgnores([
|
globalIgnores([
|
||||||
@@ -18,11 +20,11 @@ export default defineConfig([
|
|||||||
"**/*.json",
|
"**/*.json",
|
||||||
"**/.eslintrc.js.bak",
|
"**/.eslintrc.js.bak",
|
||||||
// Files from linked dependencies (those files should not exist for most people).
|
// Files from linked dependencies (those files should not exist for most people).
|
||||||
"modules/octagonal-wheels/dist/**/*",
|
"modules/octagonal-wheels/dist",
|
||||||
|
|
||||||
// Sub-projects (Exclude from root linting as they have different environments)
|
// Sub-projects (Exclude from root linting as they have different environments)
|
||||||
"src/apps/**/*",
|
"src/apps",
|
||||||
"utils/**/*",
|
"utils",
|
||||||
|
|
||||||
// Specific exclusions from common library (src/lib)
|
// Specific exclusions from common library (src/lib)
|
||||||
"src/lib/coverage",
|
"src/lib/coverage",
|
||||||
@@ -54,6 +56,7 @@ export default defineConfig([
|
|||||||
]),
|
]),
|
||||||
...sveltePlugin.configs["flat/base"],
|
...sveltePlugin.configs["flat/base"],
|
||||||
...obsidianmd.configs.recommended,
|
...obsidianmd.configs.recommended,
|
||||||
|
importAlias.configs.recommended,
|
||||||
{
|
{
|
||||||
files: ["**/*.ts"],
|
files: ["**/*.ts"],
|
||||||
// ignores:["src/lib/**/*.ts"], // Exclude library files from root linting (they have different environments and rules).
|
// ignores:["src/lib/**/*.ts"], // Exclude library files from root linting (they have different environments and rules).
|
||||||
@@ -62,64 +65,29 @@ export default defineConfig([
|
|||||||
parser: tsParser,
|
parser: tsParser,
|
||||||
parserOptions: {
|
parserOptions: {
|
||||||
project: "./tsconfig.json",
|
project: "./tsconfig.json",
|
||||||
|
rootDir: "./",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
linterOptions:{
|
linterOptions: {
|
||||||
reportUnusedDisableDirectives: false,
|
reportUnusedDisableDirectives: false,
|
||||||
},
|
},
|
||||||
rules: {
|
rules: {
|
||||||
// -- Base rules (turned off in favour of TS specific versions or explicitly disabled).
|
...baseRules,
|
||||||
"no-unused-vars": "off",
|
...obsidianRules,
|
||||||
"no-unused-labels": "off",
|
// -- Project specific rules
|
||||||
"no-prototype-builtins": "off",
|
...ImportAliasRules("."),
|
||||||
"require-await": "off",
|
|
||||||
// -- TypeScript specific rules
|
|
||||||
// @typescript-eslint/no-unsafe-* rules and @typescript-eslint/no-explicit-any:
|
|
||||||
// This project contains a lot of library-sh code where the use of `any` is often necessary and justified.
|
|
||||||
// Rules is now set to 'off' for a while.
|
|
||||||
"@typescript-eslint/no-explicit-any": "off",
|
|
||||||
"@typescript-eslint/no-unsafe-argument": "off",
|
|
||||||
"@typescript-eslint/no-unsafe-call": "off",
|
|
||||||
"@typescript-eslint/no-unsafe-member-access": "off",
|
|
||||||
"@typescript-eslint/no-unsafe-return": "off",
|
|
||||||
"@typescript-eslint/no-unsafe-assignment": "off",
|
|
||||||
// -- Reasonable rules.
|
|
||||||
"@typescript-eslint/no-deprecated": warnWhileDev,
|
|
||||||
"@typescript-eslint/no-unused-vars": ["error", { args: "none" }],
|
|
||||||
"@typescript-eslint/ban-ts-comment": "off",
|
|
||||||
"@typescript-eslint/no-empty-function": "off",
|
|
||||||
"@typescript-eslint/require-await": "error",
|
|
||||||
"@typescript-eslint/no-misused-promises": "error",
|
|
||||||
"@typescript-eslint/no-floating-promises": "error",
|
|
||||||
"@typescript-eslint/no-unnecessary-type-assertion": "error",
|
|
||||||
|
|
||||||
// -- Obsidian rules
|
|
||||||
// obsidianmd/no-unsupported-api: usually this project checks for API support at runtime, so this rule is not critical but can be helpful to catch potential issues.
|
|
||||||
"obsidianmd/no-unsupported-api": warnWhileDev,
|
|
||||||
|
|
||||||
// -- General rules
|
|
||||||
"no-async-promise-executor": warnWhileDev,
|
|
||||||
"no-constant-condition": ["error", { checkLoops: false }],
|
|
||||||
// -- Disabled rules
|
|
||||||
// no-undef: This option breaks the global declarations for the library files and is not worth the effort to fix at this time.
|
|
||||||
"no-undef": "off",
|
|
||||||
|
|
||||||
// -- Plugin specific overrides
|
|
||||||
"obsidianmd/rule-custom-message": "off",
|
|
||||||
"obsidianmd/ui/sentence-case": "off",
|
|
||||||
"obsidianmd/no-plugin-as-component": "off",
|
|
||||||
|
|
||||||
// -- Temporary overrides for migration
|
|
||||||
"obsidianmd/no-static-styles-assignment": "off",
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
files: ["**/*.svelte"],
|
files: ["**/*.svelte"],
|
||||||
languageOptions: {
|
languageOptions: {
|
||||||
|
globals: { ...globals.browser, PouchDB: "readonly" },
|
||||||
parser: svelteParser,
|
parser: svelteParser,
|
||||||
parserOptions: {
|
parserOptions: {
|
||||||
parser: tsParser,
|
parser: tsParser,
|
||||||
extraFileExtensions: [".svelte"],
|
extraFileExtensions: [".svelte"],
|
||||||
|
project: "./tsconfig.json",
|
||||||
|
rootDir: "./",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
rules: {
|
rules: {
|
||||||
@@ -127,8 +95,8 @@ export default defineConfig([
|
|||||||
// Svelte template's declarations have a lot of false positives and the rule is not worth the effort to fix at this time.
|
// Svelte template's declarations have a lot of false positives and the rule is not worth the effort to fix at this time.
|
||||||
// it may improve in the future with some options as like ["error", { argsIgnorePattern: "^_", varsIgnorePattern: "^_" }],]
|
// it may improve in the future with some options as like ["error", { argsIgnorePattern: "^_", varsIgnorePattern: "^_" }],]
|
||||||
"no-unused-vars": "off",
|
"no-unused-vars": "off",
|
||||||
"obsidianmd/no-plugin-as-component": "off",
|
...obsidianRules,
|
||||||
"obsidianmd/ui/sentence-case": "off",
|
...ImportAliasRules("."),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
]);
|
]);
|
||||||
|
|||||||
+1
-1
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"id": "obsidian-livesync",
|
"id": "obsidian-livesync",
|
||||||
"name": "Self-hosted LiveSync",
|
"name": "Self-hosted LiveSync",
|
||||||
"version": "0.25.75",
|
"version": "0.25.76",
|
||||||
"minAppVersion": "1.7.2",
|
"minAppVersion": "1.7.2",
|
||||||
"description": "Community implementation of self-hosted livesync. Reflect your vault changes to some other devices immediately. Please make sure to disable other synchronize solutions to avoid content corruption or duplication.",
|
"description": "Community implementation of self-hosted livesync. Reflect your vault changes to some other devices immediately. Please make sure to disable other synchronize solutions to avoid content corruption or duplication.",
|
||||||
"author": "vorotamoroz",
|
"author": "vorotamoroz",
|
||||||
|
|||||||
Generated
+1470
-2202
File diff suppressed because it is too large
Load Diff
+25
-22
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "obsidian-livesync",
|
"name": "obsidian-livesync",
|
||||||
"version": "0.25.75",
|
"version": "0.25.76",
|
||||||
"description": "Reflect your vault changes to some other devices immediately. Please make sure to disable other synchronize solutions to avoid content corruption or duplication.",
|
"description": "Reflect your vault changes to some other devices immediately. Please make sure to disable other synchronize solutions to avoid content corruption or duplication.",
|
||||||
"main": "main.js",
|
"main": "main.js",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
@@ -57,15 +57,17 @@
|
|||||||
"test:docker-all:stop": "npm run test:docker-all:down",
|
"test:docker-all:stop": "npm run test:docker-all:down",
|
||||||
"test:full": "npm run test:docker-all:start && vitest run --coverage && npm run test:docker-all:stop",
|
"test:full": "npm run test:docker-all:start && vitest run --coverage && npm run test:docker-all:stop",
|
||||||
"test:p2p": "bash test/suitep2p/run-p2p-tests.sh",
|
"test:p2p": "bash test/suitep2p/run-p2p-tests.sh",
|
||||||
"version": "node version-bump.mjs && git add manifest.json versions.json"
|
"update-workspaces": "node update-workspaces.mjs",
|
||||||
|
"version": "node version-bump.mjs && node update-workspaces.mjs && git add manifest.json versions.json src/apps/cli/package.json src/apps/webpeer/package.json src/apps/webapp/package.json"
|
||||||
},
|
},
|
||||||
"keywords": [],
|
"keywords": [],
|
||||||
"author": "vorotamoroz",
|
"author": "vorotamoroz",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@chialab/esbuild-plugin-worker": "^0.19.0",
|
"@dword-design/eslint-plugin-import-alias": "^8.1.8",
|
||||||
"@eslint/js": "^9.39.3",
|
"@eslint/js": "^9.39.3",
|
||||||
"@sveltejs/vite-plugin-svelte": "^6.2.4",
|
"@playwright/test": "^1.58.2",
|
||||||
|
"@sveltejs/vite-plugin-svelte": "^7.1.2",
|
||||||
"@tsconfig/svelte": "^5.0.8",
|
"@tsconfig/svelte": "^5.0.8",
|
||||||
"@types/deno": "^2.5.0",
|
"@types/deno": "^2.5.0",
|
||||||
"@types/diff-match-patch": "^1.0.36",
|
"@types/diff-match-patch": "^1.0.36",
|
||||||
@@ -80,7 +82,6 @@
|
|||||||
"@types/pouchdb-mapreduce": "^6.1.10",
|
"@types/pouchdb-mapreduce": "^6.1.10",
|
||||||
"@types/pouchdb-replication": "^6.4.7",
|
"@types/pouchdb-replication": "^6.4.7",
|
||||||
"@types/transform-pouch": "^1.0.6",
|
"@types/transform-pouch": "^1.0.6",
|
||||||
"@typescript-eslint/eslint-plugin": "8.56.1",
|
|
||||||
"@typescript-eslint/parser": "8.56.1",
|
"@typescript-eslint/parser": "8.56.1",
|
||||||
"@vitest/browser": "^4.1.8",
|
"@vitest/browser": "^4.1.8",
|
||||||
"@vitest/browser-playwright": "^4.1.8",
|
"@vitest/browser-playwright": "^4.1.8",
|
||||||
@@ -91,12 +92,11 @@
|
|||||||
"esbuild-svelte": "^0.9.4",
|
"esbuild-svelte": "^0.9.4",
|
||||||
"eslint": "^9.39.3",
|
"eslint": "^9.39.3",
|
||||||
"eslint-plugin-obsidianmd": "^0.3.0",
|
"eslint-plugin-obsidianmd": "^0.3.0",
|
||||||
"eslint-plugin-svelte": "^3.15.0",
|
"eslint-plugin-svelte": "^3.19.0",
|
||||||
"events": "^3.3.0",
|
"events": "^3.3.0",
|
||||||
"globals": "^14.0.0",
|
"globals": "^14.0.0",
|
||||||
"playwright": "^1.58.2",
|
"playwright": "^1.58.2",
|
||||||
"postcss": "^8.5.6",
|
"postcss": "^8.5.6",
|
||||||
"postcss-load-config": "^6.0.1",
|
|
||||||
"pouchdb-adapter-http": "^9.0.0",
|
"pouchdb-adapter-http": "^9.0.0",
|
||||||
"pouchdb-adapter-idb": "^9.0.0",
|
"pouchdb-adapter-idb": "^9.0.0",
|
||||||
"pouchdb-adapter-indexeddb": "^9.0.0",
|
"pouchdb-adapter-indexeddb": "^9.0.0",
|
||||||
@@ -110,20 +110,22 @@
|
|||||||
"pouchdb-utils": "^9.0.0",
|
"pouchdb-utils": "^9.0.0",
|
||||||
"prettier": "3.8.1",
|
"prettier": "3.8.1",
|
||||||
"rollup-plugin-copy": "^3.5.0",
|
"rollup-plugin-copy": "^3.5.0",
|
||||||
"svelte": "5.41.1",
|
"svelte": "5.56.3",
|
||||||
"svelte-check": "^4.4.3",
|
"svelte-check": "^4.6.0",
|
||||||
|
"svelte-eslint-parser": "^1.8.0",
|
||||||
"svelte-preprocess": "^6.0.3",
|
"svelte-preprocess": "^6.0.3",
|
||||||
"terser": "^5.39.0",
|
"terser": "^5.39.0",
|
||||||
"tinyglobby": "^0.2.15",
|
"tinyglobby": "^0.2.15",
|
||||||
"transform-pouch": "^2.0.0",
|
"transform-pouch": "^2.0.0",
|
||||||
"tslib": "^2.8.1",
|
|
||||||
"tsx": "^4.21.0",
|
"tsx": "^4.21.0",
|
||||||
"typescript": "5.9.3",
|
"typescript": "5.9.3",
|
||||||
"vite": "^7.3.1",
|
"typescript-eslint": "^8.61.0",
|
||||||
"vite-plugin-istanbul": "^8.0.0",
|
"vite": "^8.0.16",
|
||||||
"vitest": "^4.1.8",
|
"vitest": "^4.1.8",
|
||||||
"webdriverio": "^9.27.0",
|
"webdriverio": "^9.27.0",
|
||||||
"yaml": "^2.8.2"
|
"yaml": "^2.8.2",
|
||||||
|
"@emnapi/core": "1.11.1",
|
||||||
|
"@emnapi/runtime": "1.11.1"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@aws-sdk/client-s3": "^3.808.0",
|
"@aws-sdk/client-s3": "^3.808.0",
|
||||||
@@ -132,21 +134,22 @@
|
|||||||
"@smithy/middleware-apply-body-checksum": "^4.3.9",
|
"@smithy/middleware-apply-body-checksum": "^4.3.9",
|
||||||
"@smithy/protocol-http": "^5.3.9",
|
"@smithy/protocol-http": "^5.3.9",
|
||||||
"@smithy/querystring-builder": "^4.2.9",
|
"@smithy/querystring-builder": "^4.2.9",
|
||||||
|
"@smithy/types": "^4.14.3",
|
||||||
"@smithy/util-retry": "^4.4.5",
|
"@smithy/util-retry": "^4.4.5",
|
||||||
"@trystero-p2p/nostr": "^0.24.0",
|
"@trystero-p2p/nostr": "^0.24.0",
|
||||||
"chokidar": "^4.0.0",
|
|
||||||
"commander": "^14.0.3",
|
|
||||||
"diff-match-patch": "^1.0.5",
|
"diff-match-patch": "^1.0.5",
|
||||||
"fflate": "^0.8.2",
|
"fflate": "^0.8.2",
|
||||||
"idb": "^8.0.3",
|
"idb": "^8.0.3",
|
||||||
"markdown-it": "^14.1.1",
|
"markdown-it": "^14.2.0",
|
||||||
"micromatch": "^4.0.0",
|
"minimatch": "^10.2.5",
|
||||||
"minimatch": "^10.2.2",
|
"obsidian": "^1.13.1",
|
||||||
"obsidian": "^1.12.3",
|
|
||||||
"octagonal-wheels": "^0.1.46",
|
"octagonal-wheels": "^0.1.46",
|
||||||
"pouchdb-adapter-leveldb": "^9.0.0",
|
|
||||||
"qrcode-generator": "^1.4.4",
|
"qrcode-generator": "^1.4.4",
|
||||||
"werift": "^0.23.0",
|
|
||||||
"xxhash-wasm-102": "npm:xxhash-wasm@^1.0.2"
|
"xxhash-wasm-102": "npm:xxhash-wasm@^1.0.2"
|
||||||
}
|
},
|
||||||
|
"workspaces": [
|
||||||
|
"src/apps/cli",
|
||||||
|
"src/apps/webpeer",
|
||||||
|
"src/apps/webapp"
|
||||||
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -82,8 +82,8 @@ RUN apt-get update \
|
|||||||
|
|
||||||
WORKDIR /deps
|
WORKDIR /deps
|
||||||
|
|
||||||
# runtime-package.json lists only the packages that Vite leaves external
|
# package.json lists only the packages that the CLI requires
|
||||||
COPY src/apps/cli/runtime-package.json ./package.json
|
COPY src/apps/cli/package.json ./package.json
|
||||||
RUN npm install --omit=dev
|
RUN npm install --omit=dev
|
||||||
|
|
||||||
# ─────────────────────────────────────────────────────────────────────────────
|
# ─────────────────────────────────────────────────────────────────────────────
|
||||||
|
|||||||
+12
-5
@@ -118,19 +118,26 @@ git submodule update --init --recursive
|
|||||||
# Install dependencies from the repository root
|
# Install dependencies from the repository root
|
||||||
npm install
|
npm install
|
||||||
|
|
||||||
# Build the CLI from its package directory
|
# Build the CLI from the repository root
|
||||||
|
npm run build -w self-hosted-livesync-cli
|
||||||
|
|
||||||
|
# Or from the package directory
|
||||||
cd src/apps/cli
|
cd src/apps/cli
|
||||||
npm run build
|
npm run build
|
||||||
```
|
```
|
||||||
|
|
||||||
If `src/lib` is missing, `npm run build` now stops early with a targeted message
|
If `src/lib` is missing, the build process stops early with a targeted message instead of a low-level Vite `ENOENT` error.
|
||||||
instead of a low-level Vite `ENOENT` error.
|
|
||||||
|
|
||||||
Run the CLI:
|
Run the CLI:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Run with npm script (from repository root)
|
# Run with npm workspace script (from repository root)
|
||||||
npm run --silent cli -- [database-path] [command] [args...]
|
npm run cli -w self-hosted-livesync-cli -- [database-path] [command] [args...]
|
||||||
|
|
||||||
|
# Or from the package directory
|
||||||
|
cd src/apps/cli
|
||||||
|
npm run cli -- [database-path] [command] [args...]
|
||||||
|
|
||||||
# Run the built executable directly
|
# Run the built executable directly
|
||||||
node src/apps/cli/dist/index.cjs [database-path] [command] [args...]
|
node src/apps/cli/dist/index.cjs [database-path] [command] [args...]
|
||||||
```
|
```
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import * as path from "path";
|
|
||||||
import type { UXFileInfoStub, UXFolderInfo } from "@lib/common/types";
|
import type { UXFileInfoStub, UXFolderInfo } from "@lib/common/types";
|
||||||
import type { IConversionAdapter } from "@lib/serviceModules/adapters";
|
import type { IConversionAdapter } from "@lib/serviceModules/adapters";
|
||||||
import type { NodeFile, NodeFolder } from "./NodeTypes";
|
import type { NodeFile, NodeFolder } from "./NodeTypes";
|
||||||
|
import { path } from "../node-compat";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Conversion adapter implementation for Node.js
|
* Conversion adapter implementation for Node.js
|
||||||
|
|||||||
@@ -1,5 +1,3 @@
|
|||||||
import * as fs from "fs/promises";
|
|
||||||
import * as path from "path";
|
|
||||||
import type { FilePath, UXStat } from "@lib/common/types";
|
import type { FilePath, UXStat } from "@lib/common/types";
|
||||||
import type { IFileSystemAdapter } from "@lib/serviceModules/adapters";
|
import type { IFileSystemAdapter } from "@lib/serviceModules/adapters";
|
||||||
import { NodePathAdapter } from "./NodePathAdapter";
|
import { NodePathAdapter } from "./NodePathAdapter";
|
||||||
@@ -8,6 +6,7 @@ import { NodeConversionAdapter } from "./NodeConversionAdapter";
|
|||||||
import { NodeStorageAdapter } from "./NodeStorageAdapter";
|
import { NodeStorageAdapter } from "./NodeStorageAdapter";
|
||||||
import { NodeVaultAdapter } from "./NodeVaultAdapter";
|
import { NodeVaultAdapter } from "./NodeVaultAdapter";
|
||||||
import type { NodeFile, NodeFolder, NodeStat } from "./NodeTypes";
|
import type { NodeFile, NodeFolder, NodeStat } from "./NodeTypes";
|
||||||
|
import { fsPromises as fs, path } from "../node-compat";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Complete file system adapter implementation for Node.js
|
* Complete file system adapter implementation for Node.js
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import * as path from "path";
|
|
||||||
import type { FilePath } from "@lib/common/types";
|
import type { FilePath } from "@lib/common/types";
|
||||||
import type { IPathAdapter } from "@lib/serviceModules/adapters";
|
import type { IPathAdapter } from "@lib/serviceModules/adapters";
|
||||||
import type { NodeFile } from "./NodeTypes";
|
import type { NodeFile } from "./NodeTypes";
|
||||||
|
import { path } from "../node-compat";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Path adapter implementation for Node.js
|
* Path adapter implementation for Node.js
|
||||||
|
|||||||
@@ -1,8 +1,7 @@
|
|||||||
import * as fs from "fs/promises";
|
|
||||||
import * as path from "path";
|
|
||||||
import type { UXDataWriteOptions } from "@lib/common/types";
|
import type { UXDataWriteOptions } from "@lib/common/types";
|
||||||
import type { IStorageAdapter } from "@lib/serviceModules/adapters";
|
import type { IStorageAdapter } from "@lib/serviceModules/adapters";
|
||||||
import type { NodeStat } from "./NodeTypes";
|
import type { NodeStat } from "./NodeTypes";
|
||||||
|
import { fsPromises as fs, path } from "../node-compat";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Storage adapter implementation for Node.js
|
* Storage adapter implementation for Node.js
|
||||||
|
|||||||
@@ -1,8 +1,7 @@
|
|||||||
import * as fs from "fs/promises";
|
|
||||||
import * as path from "path";
|
|
||||||
import type { UXDataWriteOptions } from "@lib/common/types";
|
import type { UXDataWriteOptions } from "@lib/common/types";
|
||||||
import type { IVaultAdapter } from "@lib/serviceModules/adapters";
|
import type { IVaultAdapter } from "@lib/serviceModules/adapters";
|
||||||
import type { NodeFile, NodeFolder, NodeStat } from "./NodeTypes";
|
import type { NodeFile, NodeFolder } from "./NodeTypes";
|
||||||
|
import { fsPromises as fs, path } from "../node-compat";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Vault adapter implementation for Node.js
|
* Vault adapter implementation for Node.js
|
||||||
|
|||||||
@@ -1,15 +1,16 @@
|
|||||||
import type { LiveSyncBaseCore } from "../../../LiveSyncBaseCore";
|
import type { LiveSyncBaseCore } from "@/LiveSyncBaseCore";
|
||||||
import { P2P_DEFAULT_SETTINGS } from "@lib/common/types";
|
import { P2P_DEFAULT_SETTINGS } from "@lib/common/types";
|
||||||
import type { ServiceContext } from "@lib/services/base/ServiceBase";
|
import type { ServiceContext } from "@lib/services/base/ServiceBase";
|
||||||
import { LiveSyncTrysteroReplicator } from "@lib/replication/trystero/LiveSyncTrysteroReplicator";
|
import { LiveSyncTrysteroReplicator } from "@lib/replication/trystero/LiveSyncTrysteroReplicator";
|
||||||
import { addP2PEventHandlers } from "@lib/replication/trystero/addP2PEventHandlers";
|
import { compatGlobal } from "@lib/common/coreEnvFunctions.ts";
|
||||||
|
|
||||||
type CLIP2PPeer = {
|
type CLIP2PPeer = {
|
||||||
peerId: string;
|
peerId: string;
|
||||||
name: string;
|
name: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
function delay(ms: number): Promise<void> {
|
function delay(ms: number): Promise<void> {
|
||||||
return new Promise((resolve) => setTimeout(resolve, ms));
|
return new Promise((resolve) => compatGlobal.setTimeout(resolve, ms));
|
||||||
}
|
}
|
||||||
|
|
||||||
export function parseTimeoutSeconds(value: string, commandName: string): number {
|
export function parseTimeoutSeconds(value: string, commandName: string): number {
|
||||||
|
|||||||
@@ -1,5 +1,3 @@
|
|||||||
import * as fs from "fs/promises";
|
|
||||||
import * as path from "path";
|
|
||||||
import { decodeSettingsFromSetupURI } from "@lib/API/processSetting";
|
import { decodeSettingsFromSetupURI } from "@lib/API/processSetting";
|
||||||
import { configURIBase } from "@lib/common/models/shared.const";
|
import { configURIBase } from "@lib/common/models/shared.const";
|
||||||
import {
|
import {
|
||||||
@@ -18,6 +16,8 @@ import { promptForPassphrase, readStdinAsUtf8, toArrayBuffer, toDatabaseRelative
|
|||||||
import { collectPeers, openP2PHost, parseTimeoutSeconds, syncWithPeer } from "./p2p";
|
import { collectPeers, openP2PHost, parseTimeoutSeconds, syncWithPeer } from "./p2p";
|
||||||
import { performFullScan } from "@lib/serviceFeatures/offlineScanner";
|
import { performFullScan } from "@lib/serviceFeatures/offlineScanner";
|
||||||
import { UnresolvedErrorManager } from "@lib/services/base/UnresolvedErrorManager";
|
import { UnresolvedErrorManager } from "@lib/services/base/UnresolvedErrorManager";
|
||||||
|
import { compatGlobal } from "@lib/common/coreEnvFunctions.ts";
|
||||||
|
import { fsPromises as fs, path } from "../node-compat";
|
||||||
|
|
||||||
function redactConnectionString(uri: string): string {
|
function redactConnectionString(uri: string): string {
|
||||||
return uri.replace(/\/\/([^@/]+)@/u, "//***@");
|
return uri.replace(/\/\/([^@/]+)@/u, "//***@");
|
||||||
@@ -150,11 +150,11 @@ export async function runCommand(options: CLIOptions, context: CLICommandContext
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
pollTimer = setTimeout(poll, currentIntervalMs);
|
pollTimer = compatGlobal.setTimeout(poll, currentIntervalMs);
|
||||||
};
|
};
|
||||||
let pollTimer: ReturnType<typeof setTimeout> = setTimeout(poll, currentIntervalMs);
|
let pollTimer = compatGlobal.setTimeout(poll, currentIntervalMs);
|
||||||
core.services.appLifecycle.onUnload.addHandler(async () => {
|
core.services.appLifecycle.onUnload.addHandler(async () => {
|
||||||
clearTimeout(pollTimer);
|
compatGlobal.clearTimeout(pollTimer);
|
||||||
return true;
|
return true;
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { LiveSyncBaseCore } from "../../../LiveSyncBaseCore";
|
import { LiveSyncBaseCore } from "@/LiveSyncBaseCore";
|
||||||
import { ServiceContext } from "@lib/services/base/ServiceBase";
|
import { ServiceContext } from "@lib/services/base/ServiceBase";
|
||||||
import type { ObsidianLiveSyncSettings } from "@lib/common/types";
|
import type { ObsidianLiveSyncSettings } from "@lib/common/types";
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
import * as path from "path";
|
import { path, readline } from "../node-compat";
|
||||||
import * as readline from "node:readline/promises";
|
|
||||||
|
|
||||||
export function toArrayBuffer(data: Buffer): ArrayBuffer {
|
export function toArrayBuffer(data: Buffer): ArrayBuffer {
|
||||||
return data.buffer.slice(data.byteOffset, data.byteOffset + data.byteLength) as ArrayBuffer;
|
return data.buffer.slice(data.byteOffset, data.byteOffset + data.byteLength) as ArrayBuffer;
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
#!/usr/bin/env node
|
#!/usr/bin/env node
|
||||||
|
// eslint-disable -- This is the entry point for the CLI application.
|
||||||
import * as polyfill from "werift";
|
import * as polyfill from "werift";
|
||||||
import { main } from "./main";
|
import { main } from "./main";
|
||||||
|
|
||||||
|
|||||||
+13
-19
@@ -1,17 +1,10 @@
|
|||||||
/**
|
|
||||||
* Self-hosted LiveSync CLI
|
|
||||||
* Command-line version of Self-hosted LiveSync plugin for syncing vaults without Obsidian
|
|
||||||
*/
|
|
||||||
|
|
||||||
import * as fs from "fs/promises";
|
|
||||||
import * as path from "path";
|
|
||||||
import { NodeServiceContext, NodeServiceHub } from "./services/NodeServiceHub";
|
import { NodeServiceContext, NodeServiceHub } from "./services/NodeServiceHub";
|
||||||
import { configureNodeLocalStorage, ensureGlobalNodeLocalStorage } from "./services/NodeLocalStorage";
|
import { configureNodeLocalStorage, ensureGlobalNodeLocalStorage } from "./services/NodeLocalStorage";
|
||||||
import { LiveSyncBaseCore } from "../../LiveSyncBaseCore";
|
import { LiveSyncBaseCore } from "@/LiveSyncBaseCore";
|
||||||
import { initialiseServiceModulesCLI } from "./serviceModules/CLIServiceModules";
|
import { initialiseServiceModulesCLI } from "./serviceModules/CLIServiceModules";
|
||||||
import { DEFAULT_SETTINGS, LOG_LEVEL_VERBOSE, type LOG_LEVEL, type ObsidianLiveSyncSettings } from "@lib/common/types";
|
import { DEFAULT_SETTINGS, LOG_LEVEL_VERBOSE, type LOG_LEVEL, type ObsidianLiveSyncSettings } from "@lib/common/types";
|
||||||
import type { InjectableServiceHub } from "@lib/services/implements/injectable/InjectableServiceHub";
|
import type { InjectableServiceHub } from "@lib/services/implements/injectable/InjectableServiceHub";
|
||||||
import type { InjectableSettingService } from "@/lib/src/services/implements/injectable/InjectableSettingService";
|
import type { InjectableSettingService } from "@lib/services/implements/injectable/InjectableSettingService";
|
||||||
import {
|
import {
|
||||||
LOG_LEVEL_DEBUG,
|
LOG_LEVEL_DEBUG,
|
||||||
setGlobalLogFunction,
|
setGlobalLogFunction,
|
||||||
@@ -26,7 +19,8 @@ import type { CLICommand, CLIOptions } from "./commands/types";
|
|||||||
import { getPathFromUXFileInfo } from "@lib/common/typeUtils";
|
import { getPathFromUXFileInfo } from "@lib/common/typeUtils";
|
||||||
import { stripAllPrefixes } from "@lib/string_and_binary/path";
|
import { stripAllPrefixes } from "@lib/string_and_binary/path";
|
||||||
import { IgnoreRules } from "./serviceModules/IgnoreRules";
|
import { IgnoreRules } from "./serviceModules/IgnoreRules";
|
||||||
import { useP2PReplicatorFeature } from "@/lib/src/replication/trystero/useP2PReplicatorFeature";
|
import { useP2PReplicatorFeature } from "@lib/replication/trystero/useP2PReplicatorFeature";
|
||||||
|
import { fsPromises as fs, path, fs as fsSync } from "./node-compat";
|
||||||
|
|
||||||
const SETTINGS_FILE = ".livesync/settings.json";
|
const SETTINGS_FILE = ".livesync/settings.json";
|
||||||
ensureGlobalNodeLocalStorage();
|
ensureGlobalNodeLocalStorage();
|
||||||
@@ -238,8 +232,8 @@ async function createDefaultSettingsFile(options: CLIOptions) {
|
|||||||
const targetPath = options.settingsPath
|
const targetPath = options.settingsPath
|
||||||
? path.resolve(options.settingsPath)
|
? path.resolve(options.settingsPath)
|
||||||
: options.commandArgs[0]
|
: options.commandArgs[0]
|
||||||
? path.resolve(options.commandArgs[0])
|
? path.resolve(options.commandArgs[0])
|
||||||
: path.resolve(process.cwd(), "data.json");
|
: path.resolve(process.cwd(), "data.json");
|
||||||
|
|
||||||
if (!options.force) {
|
if (!options.force) {
|
||||||
try {
|
try {
|
||||||
@@ -329,8 +323,8 @@ export async function main() {
|
|||||||
options.command === "mirror" && options.commandArgs[0]
|
options.command === "mirror" && options.commandArgs[0]
|
||||||
? path.resolve(options.commandArgs[0])
|
? path.resolve(options.commandArgs[0])
|
||||||
: options.vaultPath
|
: options.vaultPath
|
||||||
? path.resolve(options.vaultPath)
|
? path.resolve(options.vaultPath)
|
||||||
: databasePath!;
|
: databasePath!;
|
||||||
|
|
||||||
// Check if vault directory exists
|
// Check if vault directory exists
|
||||||
try {
|
try {
|
||||||
@@ -485,8 +479,8 @@ export async function main() {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
process.on("SIGINT", () => shutdown("SIGINT"));
|
process.on("SIGINT", () => void shutdown("SIGINT"));
|
||||||
process.on("SIGTERM", () => shutdown("SIGTERM"));
|
process.on("SIGTERM", () => void shutdown("SIGTERM"));
|
||||||
|
|
||||||
// Save the settings file before any lifecycle events can mutate and persist them.
|
// Save the settings file before any lifecycle events can mutate and persist them.
|
||||||
// suspendAllSync and other lifecycle hooks clobber sync settings in memory, and
|
// suspendAllSync and other lifecycle hooks clobber sync settings in memory, and
|
||||||
@@ -499,8 +493,8 @@ export async function main() {
|
|||||||
if (settingsBackup) {
|
if (settingsBackup) {
|
||||||
const tmpPath = settingsPath + ".tmp";
|
const tmpPath = settingsPath + ".tmp";
|
||||||
try {
|
try {
|
||||||
require("fs").writeFileSync(tmpPath, settingsBackup, "utf-8");
|
fsSync.writeFileSync(tmpPath, settingsBackup, "utf-8");
|
||||||
require("fs").renameSync(tmpPath, settingsPath);
|
fsSync.renameSync(tmpPath, settingsPath);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.error("[Settings] Failed to restore settings on exit:", err);
|
console.error("[Settings] Failed to restore settings on exit:", err);
|
||||||
}
|
}
|
||||||
@@ -563,7 +557,7 @@ export async function main() {
|
|||||||
|
|
||||||
if (options.command === "daemon" && result) {
|
if (options.command === "daemon" && result) {
|
||||||
// Keep the process running
|
// Keep the process running
|
||||||
await new Promise(() => {});
|
await new Promise(() => { });
|
||||||
} else {
|
} else {
|
||||||
await core.services.control.onUnload();
|
await core.services.control.onUnload();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,12 +10,10 @@ import type {
|
|||||||
IStorageEventWatchHandlers,
|
IStorageEventWatchHandlers,
|
||||||
} from "@lib/managers/adapters";
|
} from "@lib/managers/adapters";
|
||||||
import type { FileEventItemSentinel } from "@lib/managers/StorageEventManager";
|
import type { FileEventItemSentinel } from "@lib/managers/StorageEventManager";
|
||||||
import type { NodeFile, NodeFolder } from "../adapters/NodeTypes";
|
import type { NodeFile, NodeFolder } from "@/apps/cli/adapters/NodeTypes";
|
||||||
import type { Stats } from "fs";
|
|
||||||
import * as fs from "fs/promises";
|
|
||||||
import * as path from "path";
|
|
||||||
import { watch as chokidarWatch, type FSWatcher } from "chokidar";
|
import { watch as chokidarWatch, type FSWatcher } from "chokidar";
|
||||||
import type { IgnoreRules } from "../serviceModules/IgnoreRules";
|
import type { IgnoreRules } from "@/apps/cli/serviceModules/IgnoreRules";
|
||||||
|
import { fsPromises as fs, path, type Stats } from "../node-compat";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* CLI-specific type guard adapter
|
* CLI-specific type guard adapter
|
||||||
@@ -101,7 +99,7 @@ class CLIWatchAdapter implements IStorageEventWatchAdapter {
|
|||||||
private basePath: string,
|
private basePath: string,
|
||||||
private ignoreRules?: IgnoreRules,
|
private ignoreRules?: IgnoreRules,
|
||||||
private watchEnabled: boolean = false
|
private watchEnabled: boolean = false
|
||||||
) {}
|
) { }
|
||||||
|
|
||||||
private _toNodeFile(filePath: string, stats: Stats | undefined): NodeFile {
|
private _toNodeFile(filePath: string, stats: Stats | undefined): NodeFile {
|
||||||
return {
|
return {
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { describe, expect, it, vi, beforeEach } from "vitest";
|
import { describe, expect, it, vi, beforeEach } from "vitest";
|
||||||
import type { IStorageEventWatchHandlers } from "@lib/managers/adapters";
|
import type { IStorageEventWatchHandlers } from "@lib/managers/adapters";
|
||||||
import type { NodeFile } from "../adapters/NodeTypes";
|
import type { NodeFile } from "@/apps/cli/adapters/NodeTypes";
|
||||||
|
|
||||||
// ── chokidar mock ──────────────────────────────────────────────────────────────
|
// ── chokidar mock ──────────────────────────────────────────────────────────────
|
||||||
// Must be hoisted before imports that pull in chokidar.
|
// Must be hoisted before imports that pull in chokidar.
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
import { StorageEventManagerBase, type StorageEventManagerBaseDependencies } from "@lib/managers/StorageEventManager";
|
import { StorageEventManagerBase, type StorageEventManagerBaseDependencies } from "@lib/managers/StorageEventManager";
|
||||||
import { CLIStorageEventManagerAdapter } from "./CLIStorageEventManagerAdapter";
|
import { CLIStorageEventManagerAdapter } from "./CLIStorageEventManagerAdapter";
|
||||||
import type { IMinimumLiveSyncCommands, LiveSyncBaseCore } from "../../../LiveSyncBaseCore";
|
import type { IMinimumLiveSyncCommands, LiveSyncBaseCore } from "@/LiveSyncBaseCore";
|
||||||
import type { ServiceContext } from "@lib/services/base/ServiceBase";
|
import type { ServiceContext } from "@lib/services/base/ServiceBase";
|
||||||
import type { IgnoreRules } from "../serviceModules/IgnoreRules";
|
import type { IgnoreRules } from "@/apps/cli/serviceModules/IgnoreRules";
|
||||||
// import type { IMinimumLiveSyncCommands } from "@lib/services/base/IService";
|
// import type { IMinimumLiveSyncCommands } from "@lib/services/base/IService";
|
||||||
|
|
||||||
export class StorageEventManagerCLI extends StorageEventManagerBase<CLIStorageEventManagerAdapter> {
|
export class StorageEventManagerCLI extends StorageEventManagerBase<CLIStorageEventManagerAdapter> {
|
||||||
|
|||||||
@@ -0,0 +1,13 @@
|
|||||||
|
/* eslint-disable obsidianmd/no-nodejs-builtins */
|
||||||
|
import * as nodeFs from "node:fs";
|
||||||
|
import * as nodeFsPromises from "node:fs/promises";
|
||||||
|
import * as nodePath from "node:path";
|
||||||
|
import * as nodeReadlinePromises from "node:readline/promises";
|
||||||
|
import type { Stats } from "node:fs";
|
||||||
|
export {
|
||||||
|
nodeFs as fs,
|
||||||
|
nodeFsPromises as fsPromises,
|
||||||
|
nodePath as path,
|
||||||
|
nodeReadlinePromises as readline,
|
||||||
|
type Stats,
|
||||||
|
};
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "self-hosted-livesync-cli",
|
"name": "self-hosted-livesync-cli",
|
||||||
"private": true,
|
"private": true,
|
||||||
"version": "0.0.0",
|
"version": "0.25.76-cli",
|
||||||
"main": "dist/index.cjs",
|
"main": "dist/index.cjs",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
@@ -38,6 +38,26 @@
|
|||||||
"test:e2e:docker:p2p-sync": "RUN_BUILD=0 LIVESYNC_TEST_DOCKER=1 bash test/test-p2p-sync-linux.sh",
|
"test:e2e:docker:p2p-sync": "RUN_BUILD=0 LIVESYNC_TEST_DOCKER=1 bash test/test-p2p-sync-linux.sh",
|
||||||
"test:e2e:docker:all": "export RUN_BUILD=0 && npm run test:e2e:docker:setup-put-cat && npm run test:e2e:docker:push-pull && npm run test:e2e:docker:sync-two-local && npm run test:e2e:docker:mirror && npm run test:e2e:docker:remote-commands"
|
"test:e2e:docker:all": "export RUN_BUILD=0 && npm run test:e2e:docker:setup-put-cat && npm run test:e2e:docker:push-pull && npm run test:e2e:docker:sync-two-local && npm run test:e2e:docker:mirror && npm run test:e2e:docker:remote-commands"
|
||||||
},
|
},
|
||||||
"dependencies": {},
|
"dependencies": {
|
||||||
"devDependencies": {}
|
"chokidar": "^4.0.0",
|
||||||
|
"minimatch": "^10.2.5",
|
||||||
|
"octagonal-wheels": "^0.1.46",
|
||||||
|
"pouchdb-adapter-http": "^9.0.0",
|
||||||
|
"pouchdb-adapter-leveldb": "^9.0.0",
|
||||||
|
"pouchdb-core": "^9.0.0",
|
||||||
|
"pouchdb-errors": "^9.0.0",
|
||||||
|
"pouchdb-find": "^9.0.0",
|
||||||
|
"pouchdb-mapreduce": "^9.0.0",
|
||||||
|
"pouchdb-merge": "^9.0.0",
|
||||||
|
"pouchdb-replication": "^9.0.0",
|
||||||
|
"pouchdb-utils": "^9.0.0",
|
||||||
|
"transform-pouch": "^2.0.0",
|
||||||
|
"werift": "^0.23.0"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@sveltejs/vite-plugin-svelte": "^7.1.2",
|
||||||
|
"typescript": "5.9.3",
|
||||||
|
"vite": "^8.0.16",
|
||||||
|
"vitest": "^4.1.8"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,25 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "livesync-cli-runtime",
|
|
||||||
"private": true,
|
|
||||||
"version": "0.0.0",
|
|
||||||
"description": "Runtime dependencies for Self-hosted LiveSync CLI Docker image",
|
|
||||||
"dependencies": {
|
|
||||||
"chokidar": "^4.0.0",
|
|
||||||
"commander": "^14.0.3",
|
|
||||||
"werift": "^0.22.9",
|
|
||||||
"pouchdb-adapter-http": "^9.0.0",
|
|
||||||
"pouchdb-adapter-idb": "^9.0.0",
|
|
||||||
"pouchdb-adapter-indexeddb": "^9.0.0",
|
|
||||||
"pouchdb-adapter-leveldb": "^9.0.0",
|
|
||||||
"pouchdb-adapter-memory": "^9.0.0",
|
|
||||||
"pouchdb-core": "^9.0.0",
|
|
||||||
"pouchdb-errors": "^9.0.0",
|
|
||||||
"pouchdb-find": "^9.0.0",
|
|
||||||
"pouchdb-mapreduce": "^9.0.0",
|
|
||||||
"pouchdb-merge": "^9.0.0",
|
|
||||||
"pouchdb-replication": "^9.0.0",
|
|
||||||
"pouchdb-utils": "^9.0.0",
|
|
||||||
"pouchdb-wrappers": "*",
|
|
||||||
"transform-pouch": "^2.0.0"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,13 +1,13 @@
|
|||||||
import type { InjectableServiceHub } from "@lib/services/implements/injectable/InjectableServiceHub";
|
import type { InjectableServiceHub } from "@lib/services/implements/injectable/InjectableServiceHub";
|
||||||
import { ServiceRebuilder } from "@lib/serviceModules/Rebuilder";
|
import { ServiceRebuilder } from "@lib/serviceModules/Rebuilder";
|
||||||
import { ServiceFileHandler } from "../../../serviceModules/FileHandler";
|
import { ServiceFileHandler } from "@/serviceModules/FileHandler";
|
||||||
import { StorageAccessManager } from "@lib/managers/StorageProcessingManager";
|
import { StorageAccessManager } from "@lib/managers/StorageProcessingManager";
|
||||||
import type { LiveSyncBaseCore } from "../../../LiveSyncBaseCore";
|
import type { LiveSyncBaseCore } from "@/LiveSyncBaseCore";
|
||||||
import type { ServiceContext } from "@lib/services/base/ServiceBase";
|
import type { ServiceContext } from "@lib/services/base/ServiceBase";
|
||||||
import { FileAccessCLI } from "./FileAccessCLI";
|
import { FileAccessCLI } from "./FileAccessCLI";
|
||||||
import { ServiceFileAccessCLI } from "./ServiceFileAccessImpl";
|
import { ServiceFileAccessCLI } from "./ServiceFileAccessImpl";
|
||||||
import { ServiceDatabaseFileAccessCLI } from "./DatabaseFileAccess";
|
import { ServiceDatabaseFileAccessCLI } from "./DatabaseFileAccess";
|
||||||
import { StorageEventManagerCLI } from "../managers/StorageEventManagerCLI";
|
import { StorageEventManagerCLI } from "@/apps/cli/managers/StorageEventManagerCLI";
|
||||||
import type { ServiceModules } from "@lib/interfaces/ServiceModule";
|
import type { ServiceModules } from "@lib/interfaces/ServiceModule";
|
||||||
import type { IgnoreRules } from "./IgnoreRules";
|
import type { IgnoreRules } from "./IgnoreRules";
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { FileAccessBase, type FileAccessBaseDependencies } from "@lib/serviceModules/FileAccessBase";
|
import { FileAccessBase, type FileAccessBaseDependencies } from "@lib/serviceModules/FileAccessBase";
|
||||||
import { NodeFileSystemAdapter } from "../adapters/NodeFileSystemAdapter";
|
import { NodeFileSystemAdapter } from "@/apps/cli/adapters/NodeFileSystemAdapter";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* CLI-specific implementation of FileAccessBase
|
* CLI-specific implementation of FileAccessBase
|
||||||
|
|||||||
@@ -1,7 +1,5 @@
|
|||||||
import * as fs from "fs/promises";
|
|
||||||
import * as path from "path";
|
|
||||||
|
|
||||||
import { minimatch } from "minimatch";
|
import { minimatch } from "minimatch";
|
||||||
|
import { fsPromises as fs, path } from "../node-compat";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Loads and evaluates ignore rules from `.livesync/ignore` inside the vault.
|
* Loads and evaluates ignore rules from `.livesync/ignore` inside the vault.
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { ServiceFileAccessBase, type StorageAccessBaseDependencies } from "@lib/serviceModules/ServiceFileAccessBase";
|
import { ServiceFileAccessBase, type StorageAccessBaseDependencies } from "@lib/serviceModules/ServiceFileAccessBase";
|
||||||
import { NodeFileSystemAdapter } from "../adapters/NodeFileSystemAdapter";
|
import { NodeFileSystemAdapter } from "@/apps/cli/adapters/NodeFileSystemAdapter";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* CLI-specific implementation of ServiceFileAccess
|
* CLI-specific implementation of ServiceFileAccess
|
||||||
|
|||||||
@@ -7,8 +7,7 @@ import type { InjectableDatabaseEventService } from "@lib/services/implements/in
|
|||||||
import type { IVaultService } from "@lib/services/base/IService";
|
import type { IVaultService } from "@lib/services/base/IService";
|
||||||
import type { SimpleStore } from "octagonal-wheels/databases/SimpleStoreBase";
|
import type { SimpleStore } from "octagonal-wheels/databases/SimpleStoreBase";
|
||||||
import { createInstanceLogFunction } from "@lib/services/lib/logUtils";
|
import { createInstanceLogFunction } from "@lib/services/lib/logUtils";
|
||||||
import * as nodeFs from "node:fs";
|
import { fs as nodeFs, path as nodePath } from "../node-compat";
|
||||||
import * as nodePath from "node:path";
|
|
||||||
|
|
||||||
const NODE_KV_TYPED_KEY = "__nodeKvType";
|
const NODE_KV_TYPED_KEY = "__nodeKvType";
|
||||||
const NODE_KV_VALUES_KEY = "values";
|
const NODE_KV_VALUES_KEY = "values";
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
import * as nodeFs from "node:fs";
|
import { fs as nodeFs, path as nodePath } from "../node-compat";
|
||||||
import * as nodePath from "node:path";
|
|
||||||
|
|
||||||
type LocalStorageShape = {
|
type LocalStorageShape = {
|
||||||
getItem(key: string): string | null;
|
getItem(key: string): string | null;
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
import type { AppLifecycleService, AppLifecycleServiceDependencies } from "@lib/services/base/AppLifecycleService";
|
import type { AppLifecycleService, AppLifecycleServiceDependencies } from "@lib/services/base/AppLifecycleService";
|
||||||
import { ServiceContext } from "@lib/services/base/ServiceBase";
|
import { ServiceContext } from "@lib/services/base/ServiceBase";
|
||||||
import * as nodePath from "node:path";
|
|
||||||
import { ConfigServiceBrowserCompat } from "@lib/services/implements/browser/ConfigServiceBrowserCompat";
|
import { ConfigServiceBrowserCompat } from "@lib/services/implements/browser/ConfigServiceBrowserCompat";
|
||||||
import { SvelteDialogManagerBase, type ComponentHasResult } from "@lib/services/implements/base/SvelteDialog";
|
import { SvelteDialogManagerBase, type ComponentHasResult } from "@lib/services/implements/base/SvelteDialog";
|
||||||
import { UIService } from "@lib/services/implements/base/UIService";
|
import { UIService } from "@lib/services/implements/base/UIService";
|
||||||
@@ -24,7 +23,8 @@ import type { ServiceInstances } from "@lib/services/ServiceHub";
|
|||||||
import { NodeKeyValueDBService } from "./NodeKeyValueDBService";
|
import { NodeKeyValueDBService } from "./NodeKeyValueDBService";
|
||||||
import { NodeSettingService } from "./NodeSettingService";
|
import { NodeSettingService } from "./NodeSettingService";
|
||||||
import { DatabaseService } from "@lib/services/base/DatabaseService";
|
import { DatabaseService } from "@lib/services/base/DatabaseService";
|
||||||
import type { ObsidianLiveSyncSettings } from "@/lib/src/common/types";
|
import type { ObsidianLiveSyncSettings } from "@lib/common/types";
|
||||||
|
import { path as nodePath } from "../node-compat";
|
||||||
|
|
||||||
export class NodeServiceContext extends ServiceContext {
|
export class NodeServiceContext extends ServiceContext {
|
||||||
databasePath: string;
|
databasePath: string;
|
||||||
|
|||||||
@@ -77,7 +77,9 @@ export class BackgroundCliProcess {
|
|||||||
if (this.combined.includes(needle)) return;
|
if (this.combined.includes(needle)) return;
|
||||||
const status = await Promise.race([
|
const status = await Promise.race([
|
||||||
this.child.status.then((s) => ({ type: "status" as const, status: s })),
|
this.child.status.then((s) => ({ type: "status" as const, status: s })),
|
||||||
new Promise<{ type: "tick" }>((resolve) => setTimeout(() => resolve({ type: "tick" }), 100)),
|
new Promise<{ type: "tick" }>((resolve) =>
|
||||||
|
setTimeout(() => resolve({ type: "tick" }), 100)
|
||||||
|
),
|
||||||
]);
|
]);
|
||||||
if (status.type === "status") {
|
if (status.type === "status") {
|
||||||
throw new Error(
|
throw new Error(
|
||||||
|
|||||||
@@ -109,7 +109,11 @@ Deno.test("daemon: ignore rules behaviour", async (t) => {
|
|||||||
await runCliOrFail(vaultDir, "--settings", settingsFile, "mirror");
|
await runCliOrFail(vaultDir, "--settings", settingsFile, "mirror");
|
||||||
|
|
||||||
const dbList = await runCliOrFail(vaultDir, "--settings", settingsFile, "ls");
|
const dbList = await runCliOrFail(vaultDir, "--settings", settingsFile, "ls");
|
||||||
assertNotContains(dbList, "debug.log", "debug.log (ignored via .gitignore import) was unexpectedly synced to database");
|
assertNotContains(
|
||||||
|
dbList,
|
||||||
|
"debug.log",
|
||||||
|
"debug.log (ignored via .gitignore import) was unexpectedly synced to database"
|
||||||
|
);
|
||||||
assertContains(dbList, "regular.md", "regular.md was not synced normally alongside .gitignore import rules");
|
assertContains(dbList, "regular.md", "regular.md was not synced normally alongside .gitignore import rules");
|
||||||
console.log("[PASS] Case 3 verified successfully");
|
console.log("[PASS] Case 3 verified successfully");
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -46,9 +46,7 @@ Deno.test("decoupled database and vault", async () => {
|
|||||||
console.log("[INFO] applying CouchDB environment variables to settings");
|
console.log("[INFO] applying CouchDB environment variables to settings");
|
||||||
await applyCouchdbSettings(settingsFile, uri, user, password, dbname);
|
await applyCouchdbSettings(settingsFile, uri, user, password, dbname);
|
||||||
} else {
|
} else {
|
||||||
console.warn(
|
console.warn("[WARN] CouchDB environment variables are not fully set. Push and pull operations may fail.");
|
||||||
"[WARN] CouchDB environment variables are not fully set. Push and pull operations may fail."
|
|
||||||
);
|
|
||||||
await markSettingsConfigured(settingsFile);
|
await markSettingsConfigured(settingsFile);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -59,29 +57,11 @@ Deno.test("decoupled database and vault", async () => {
|
|||||||
|
|
||||||
// 1. Test push command with decoupled vault directory
|
// 1. Test push command with decoupled vault directory
|
||||||
console.log(`[INFO] push with decoupled vault -> ${REMOTE_PATH}`);
|
console.log(`[INFO] push with decoupled vault -> ${REMOTE_PATH}`);
|
||||||
await runCliOrFail(
|
await runCliOrFail(dbDir, "--vault", vaultDir, "--settings", settingsFile, "push", srcFile, REMOTE_PATH);
|
||||||
dbDir,
|
|
||||||
"--vault",
|
|
||||||
vaultDir,
|
|
||||||
"--settings",
|
|
||||||
settingsFile,
|
|
||||||
"push",
|
|
||||||
srcFile,
|
|
||||||
REMOTE_PATH
|
|
||||||
);
|
|
||||||
|
|
||||||
// 2. Test pull command with decoupled vault directory
|
// 2. Test pull command with decoupled vault directory
|
||||||
console.log(`[INFO] pull with decoupled vault <- ${REMOTE_PATH}`);
|
console.log(`[INFO] pull with decoupled vault <- ${REMOTE_PATH}`);
|
||||||
await runCliOrFail(
|
await runCliOrFail(dbDir, "--vault", vaultDir, "--settings", settingsFile, "pull", REMOTE_PATH, pulledFile);
|
||||||
dbDir,
|
|
||||||
"--vault",
|
|
||||||
vaultDir,
|
|
||||||
"--settings",
|
|
||||||
settingsFile,
|
|
||||||
"pull",
|
|
||||||
REMOTE_PATH,
|
|
||||||
pulledFile
|
|
||||||
);
|
|
||||||
|
|
||||||
const pulled = await Deno.readTextFile(pulledFile);
|
const pulled = await Deno.readTextFile(pulledFile);
|
||||||
assertEquals(pulled, content, "push/pull roundtrip with decoupled vault content mismatch");
|
assertEquals(pulled, content, "push/pull roundtrip with decoupled vault content mismatch");
|
||||||
@@ -93,14 +73,7 @@ Deno.test("decoupled database and vault", async () => {
|
|||||||
|
|
||||||
// 4. Test mirror command with decoupled vault directory
|
// 4. Test mirror command with decoupled vault directory
|
||||||
console.log("[INFO] mirror with decoupled vault");
|
console.log("[INFO] mirror with decoupled vault");
|
||||||
await runCliOrFail(
|
await runCliOrFail(dbDir, "--vault", vaultDir, "--settings", settingsFile, "mirror");
|
||||||
dbDir,
|
|
||||||
"--vault",
|
|
||||||
vaultDir,
|
|
||||||
"--settings",
|
|
||||||
settingsFile,
|
|
||||||
"mirror"
|
|
||||||
);
|
|
||||||
|
|
||||||
const restoredFile = join(vaultDir, REMOTE_PATH);
|
const restoredFile = join(vaultDir, REMOTE_PATH);
|
||||||
const restored = await Deno.readTextFile(restoredFile);
|
const restored = await Deno.readTextFile(restoredFile);
|
||||||
|
|||||||
@@ -61,11 +61,7 @@ Deno.test("remote management commands", async () => {
|
|||||||
// 1. remote-status outputs valid JSON with CouchDB details
|
// 1. remote-status outputs valid JSON with CouchDB details
|
||||||
console.log("[CASE] remote-status outputs valid JSON with CouchDB details");
|
console.log("[CASE] remote-status outputs valid JSON with CouchDB details");
|
||||||
const statusOutput = await runCliCombinedOrFail(vaultDir, "--settings", settingsFile, "remote-status");
|
const statusOutput = await runCliCombinedOrFail(vaultDir, "--settings", settingsFile, "remote-status");
|
||||||
assertContains(
|
assertContains(statusOutput, `"db_name": "${dbname}"`, "remote-status should return JSON containing db_name");
|
||||||
statusOutput,
|
|
||||||
`"db_name": "${dbname}"`,
|
|
||||||
"remote-status should return JSON containing db_name"
|
|
||||||
);
|
|
||||||
console.log("[PASS] remote-status verified");
|
console.log("[PASS] remote-status verified");
|
||||||
|
|
||||||
// 2. lock-remote locks and verifies state
|
// 2. lock-remote locks and verifies state
|
||||||
|
|||||||
@@ -132,7 +132,7 @@ Deno.test("CLI file operations: push / cat / ls / info / rm / resolve / cat-rev
|
|||||||
assertEquals(data.path, REMOTE_PATH, "info .path mismatch");
|
assertEquals(data.path, REMOTE_PATH, "info .path mismatch");
|
||||||
assertEquals(data.filename, REMOTE_PATH.split("/").at(-1), "info .filename mismatch");
|
assertEquals(data.filename, REMOTE_PATH.split("/").at(-1), "info .filename mismatch");
|
||||||
assert(typeof data.size === "number" && data.size >= 0, `info .size invalid: ${data.size}`);
|
assert(typeof data.size === "number" && data.size >= 0, `info .size invalid: ${data.size}`);
|
||||||
assert(typeof data.chunks === "number" && (data.chunks as number) >= 1, `info .chunks invalid: ${data.chunks}`);
|
assert(typeof data.chunks === "number" && (data.chunks) >= 1, `info .chunks invalid: ${data.chunks}`);
|
||||||
assertEquals(data.conflicts, "N/A", "info .conflicts should be N/A");
|
assertEquals(data.conflicts, "N/A", "info .conflicts should be N/A");
|
||||||
console.log("[PASS] info output format matched");
|
console.log("[PASS] info output format matched");
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -15,18 +15,18 @@
|
|||||||
"noEmit": true,
|
"noEmit": true,
|
||||||
|
|
||||||
/* Linting */
|
/* Linting */
|
||||||
"strict": false,
|
"strict": true,
|
||||||
|
// "noImplicitAny": false,
|
||||||
"noUnusedLocals": false,
|
"noUnusedLocals": false,
|
||||||
"noUnusedParameters": false,
|
"noUnusedParameters": false,
|
||||||
"noFallthroughCasesInSwitch": true,
|
"noFallthroughCasesInSwitch": true,
|
||||||
|
// "rootDir": "../../../",
|
||||||
/* Path mapping */
|
/* Path mapping */
|
||||||
"baseUrl": ".",
|
|
||||||
"paths": {
|
"paths": {
|
||||||
"@/*": ["../../*"],
|
"@/*": ["../../*"],
|
||||||
"@lib/*": ["../../lib/src/*"]
|
"@lib/*": ["../../lib/src/*"]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"include": ["*.ts", "**/*.ts", "**/*.tsx"],
|
"include": ["*.ts", "**/*.ts", "**/*.tsx"],
|
||||||
"exclude": ["node_modules", "dist"]
|
"exclude": ["node_modules", "dist", "test", "testdeno"]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -35,8 +35,15 @@ npm install
|
|||||||
|
|
||||||
### Development
|
### Development
|
||||||
|
|
||||||
|
From the repository root:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
npm run dev -w livesync-webapp
|
||||||
|
```
|
||||||
|
|
||||||
|
Or from the package directory:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Build the project (ensure you are in `src/apps/webapp` directory)
|
|
||||||
cd src/apps/webapp
|
cd src/apps/webapp
|
||||||
npm run dev
|
npm run dev
|
||||||
```
|
```
|
||||||
@@ -45,8 +52,15 @@ This will start a development server at `http://localhost:3000`.
|
|||||||
|
|
||||||
### Build
|
### Build
|
||||||
|
|
||||||
|
From the repository root:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
npm run build -w livesync-webapp
|
||||||
|
```
|
||||||
|
|
||||||
|
Or from the package directory:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Build the project (ensure you are in `src/apps/webapp` directory)
|
|
||||||
cd src/apps/webapp
|
cd src/apps/webapp
|
||||||
npm run build
|
npm run build
|
||||||
```
|
```
|
||||||
|
|||||||
@@ -1,11 +1,12 @@
|
|||||||
import { LiveSyncWebApp } from "./main";
|
import { LiveSyncWebApp } from "./main";
|
||||||
import { VaultHistoryStore, type VaultHistoryItem } from "./vaultSelector";
|
import { VaultHistoryStore, type VaultHistoryItem } from "./vaultSelector";
|
||||||
|
import { compatGlobal, _activeDocument } from "@lib/common/coreEnvFunctions.ts";
|
||||||
|
|
||||||
const historyStore = new VaultHistoryStore();
|
const historyStore = new VaultHistoryStore();
|
||||||
let app: LiveSyncWebApp | null = null;
|
let app: LiveSyncWebApp | null = null;
|
||||||
|
|
||||||
function getRequiredElement<T extends HTMLElement>(id: string): T {
|
function getRequiredElement<T extends HTMLElement>(id: string): T {
|
||||||
const element = document.getElementById(id);
|
const element = _activeDocument.getElementById(id);
|
||||||
if (!element) {
|
if (!element) {
|
||||||
throw new Error(`Missing element: #${id}`);
|
throw new Error(`Missing element: #${id}`);
|
||||||
}
|
}
|
||||||
@@ -22,7 +23,7 @@ function setBusyState(isBusy: boolean): void {
|
|||||||
const pickNewBtn = getRequiredElement<HTMLButtonElement>("pick-new-vault");
|
const pickNewBtn = getRequiredElement<HTMLButtonElement>("pick-new-vault");
|
||||||
pickNewBtn.disabled = isBusy;
|
pickNewBtn.disabled = isBusy;
|
||||||
|
|
||||||
const historyButtons = document.querySelectorAll<HTMLButtonElement>(".vault-item button");
|
const historyButtons = _activeDocument.querySelectorAll<HTMLButtonElement>(".vault-item button");
|
||||||
historyButtons.forEach((button) => {
|
historyButtons.forEach((button) => {
|
||||||
button.disabled = isBusy;
|
button.disabled = isBusy;
|
||||||
});
|
});
|
||||||
@@ -45,24 +46,24 @@ async function renderHistoryList(): Promise<VaultHistoryItem[]> {
|
|||||||
emptyEl.classList.toggle("is-hidden", items.length > 0);
|
emptyEl.classList.toggle("is-hidden", items.length > 0);
|
||||||
|
|
||||||
for (const item of items) {
|
for (const item of items) {
|
||||||
const row = document.createElement("div");
|
const row = _activeDocument.createElement("div");
|
||||||
row.className = "vault-item";
|
row.className = "vault-item";
|
||||||
|
|
||||||
const info = document.createElement("div");
|
const info = _activeDocument.createElement("div");
|
||||||
info.className = "vault-item-info";
|
info.className = "vault-item-info";
|
||||||
|
|
||||||
const name = document.createElement("div");
|
const name = _activeDocument.createElement("div");
|
||||||
name.className = "vault-item-name";
|
name.className = "vault-item-name";
|
||||||
name.textContent = item.name;
|
name.textContent = item.name;
|
||||||
|
|
||||||
const meta = document.createElement("div");
|
const meta = _activeDocument.createElement("div");
|
||||||
meta.className = "vault-item-meta";
|
meta.className = "vault-item-meta";
|
||||||
const label = item.id === lastUsedId ? "Last used" : "Used";
|
const label = item.id === lastUsedId ? "Last used" : "Used";
|
||||||
meta.textContent = `${label}: ${formatLastUsed(item.lastUsedAt)}`;
|
meta.textContent = `${label}: ${formatLastUsed(item.lastUsedAt)}`;
|
||||||
|
|
||||||
info.append(name, meta);
|
info.append(name, meta);
|
||||||
|
|
||||||
const useButton = document.createElement("button");
|
const useButton = _activeDocument.createElement("button");
|
||||||
useButton.type = "button";
|
useButton.type = "button";
|
||||||
useButton.textContent = "Use this vault";
|
useButton.textContent = "Use this vault";
|
||||||
useButton.addEventListener("click", () => {
|
useButton.addEventListener("click", () => {
|
||||||
@@ -120,7 +121,7 @@ async function initializeVaultSelector(): Promise<void> {
|
|||||||
await renderHistoryList();
|
await renderHistoryList();
|
||||||
}
|
}
|
||||||
|
|
||||||
window.addEventListener("load", async () => {
|
compatGlobal.addEventListener("load", async () => {
|
||||||
try {
|
try {
|
||||||
await initializeVaultSelector();
|
await initializeVaultSelector();
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
@@ -129,11 +130,11 @@ window.addEventListener("load", async () => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
window.addEventListener("beforeunload", () => {
|
compatGlobal.addEventListener("beforeunload", () => {
|
||||||
void app?.shutdown();
|
void app?.shutdown();
|
||||||
});
|
});
|
||||||
|
|
||||||
(window as any).livesyncApp = {
|
(compatGlobal as any).livesyncApp = {
|
||||||
getApp: () => app,
|
getApp: () => app,
|
||||||
historyStore,
|
historyStore,
|
||||||
};
|
};
|
||||||
|
|||||||
+10
-9
@@ -17,8 +17,9 @@ import { useSetupURIFeature } from "@lib/serviceFeatures/setupObsidian/setupUri"
|
|||||||
import { useRemoteConfiguration } from "@lib/serviceFeatures/remoteConfig";
|
import { useRemoteConfiguration } from "@lib/serviceFeatures/remoteConfig";
|
||||||
import { SetupManager } from "@/modules/features/SetupManager";
|
import { SetupManager } from "@/modules/features/SetupManager";
|
||||||
import { useSetupManagerHandlersFeature } from "@/serviceFeatures/setupObsidian/setupManagerHandlers";
|
import { useSetupManagerHandlersFeature } from "@/serviceFeatures/setupObsidian/setupManagerHandlers";
|
||||||
import { useP2PReplicatorCommands } from "@/lib/src/replication/trystero/useP2PReplicatorCommands";
|
import { useP2PReplicatorCommands } from "@lib/replication/trystero/useP2PReplicatorCommands";
|
||||||
import { useP2PReplicatorFeature } from "@/lib/src/replication/trystero/useP2PReplicatorFeature";
|
import { useP2PReplicatorFeature } from "@lib/replication/trystero/useP2PReplicatorFeature";
|
||||||
|
import { compatGlobal, _activeDocument } from "@lib/common/coreEnvFunctions.ts";
|
||||||
|
|
||||||
const SETTINGS_DIR = ".livesync";
|
const SETTINGS_DIR = ".livesync";
|
||||||
const SETTINGS_FILE = "settings.json";
|
const SETTINGS_FILE = "settings.json";
|
||||||
@@ -91,7 +92,7 @@ class LiveSyncWebApp {
|
|||||||
console.log("[Settings] Loaded from .livesync/settings.json");
|
console.log("[Settings] Loaded from .livesync/settings.json");
|
||||||
return { ...DEFAULT_SETTINGS, ...data } as ObsidianLiveSyncSettings;
|
return { ...DEFAULT_SETTINGS, ...data } as ObsidianLiveSyncSettings;
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch {
|
||||||
console.log("[Settings] Failed to load, using defaults");
|
console.log("[Settings] Failed to load, using defaults");
|
||||||
}
|
}
|
||||||
return DEFAULT_SETTINGS as ObsidianLiveSyncSettings;
|
return DEFAULT_SETTINGS as ObsidianLiveSyncSettings;
|
||||||
@@ -102,8 +103,8 @@ class LiveSyncWebApp {
|
|||||||
console.log("[AppLifecycle] Restart requested");
|
console.log("[AppLifecycle] Restart requested");
|
||||||
await this.shutdown();
|
await this.shutdown();
|
||||||
await this.initialize();
|
await this.initialize();
|
||||||
setTimeout(() => {
|
compatGlobal.setTimeout(() => {
|
||||||
window.location.reload();
|
compatGlobal.location.reload();
|
||||||
}, 1000);
|
}, 1000);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -169,7 +170,7 @@ class LiveSyncWebApp {
|
|||||||
const file = await fileHandle.getFile();
|
const file = await fileHandle.getFile();
|
||||||
const text = await file.text();
|
const text = await file.text();
|
||||||
return JSON.parse(text);
|
return JSON.parse(text);
|
||||||
} catch (error) {
|
} catch {
|
||||||
// File doesn't exist yet
|
// File doesn't exist yet
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@@ -235,7 +236,7 @@ class LiveSyncWebApp {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private showError(message: string) {
|
private showError(message: string) {
|
||||||
const statusEl = document.getElementById("status");
|
const statusEl = _activeDocument.getElementById("status");
|
||||||
if (statusEl) {
|
if (statusEl) {
|
||||||
statusEl.className = "error";
|
statusEl.className = "error";
|
||||||
statusEl.textContent = `Error: ${message}`;
|
statusEl.textContent = `Error: ${message}`;
|
||||||
@@ -243,7 +244,7 @@ class LiveSyncWebApp {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private showWarning(message: string) {
|
private showWarning(message: string) {
|
||||||
const statusEl = document.getElementById("status");
|
const statusEl = _activeDocument.getElementById("status");
|
||||||
if (statusEl) {
|
if (statusEl) {
|
||||||
statusEl.className = "warning";
|
statusEl.className = "warning";
|
||||||
statusEl.textContent = `Warning: ${message}`;
|
statusEl.textContent = `Warning: ${message}`;
|
||||||
@@ -251,7 +252,7 @@ class LiveSyncWebApp {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private showSuccess(message: string) {
|
private showSuccess(message: string) {
|
||||||
const statusEl = document.getElementById("status");
|
const statusEl = _activeDocument.getElementById("status");
|
||||||
if (statusEl) {
|
if (statusEl) {
|
||||||
statusEl.className = "success";
|
statusEl.className = "success";
|
||||||
statusEl.textContent = message;
|
statusEl.textContent = message;
|
||||||
|
|||||||
@@ -10,7 +10,8 @@ import type {
|
|||||||
IStorageEventWatchHandlers,
|
IStorageEventWatchHandlers,
|
||||||
} from "@lib/managers/adapters";
|
} from "@lib/managers/adapters";
|
||||||
import type { FileEventItemSentinel } from "@lib/managers/StorageEventManager";
|
import type { FileEventItemSentinel } from "@lib/managers/StorageEventManager";
|
||||||
import type { FSAPIFile, FSAPIFolder } from "../adapters/FSAPITypes";
|
import type { FSAPIFile, FSAPIFolder } from "@/apps/webapp/adapters/FSAPITypes";
|
||||||
|
import { compatGlobal } from "@lib/common/coreEnvFunctions.ts";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* FileSystem API-specific type guard adapter
|
* FileSystem API-specific type guard adapter
|
||||||
@@ -149,14 +150,14 @@ class FSAPIWatchAdapter implements IStorageEventWatchAdapter {
|
|||||||
|
|
||||||
async beginWatch(handlers: IStorageEventWatchHandlers): Promise<void> {
|
async beginWatch(handlers: IStorageEventWatchHandlers): Promise<void> {
|
||||||
// Use FileSystemObserver if available (Chrome 124+)
|
// Use FileSystemObserver if available (Chrome 124+)
|
||||||
if (typeof (window as any).FileSystemObserver === "undefined") {
|
if (typeof (compatGlobal as any).FileSystemObserver === "undefined") {
|
||||||
console.log("[FSAPIWatchAdapter] FileSystemObserver not available, file watching disabled");
|
console.log("[FSAPIWatchAdapter] FileSystemObserver not available, file watching disabled");
|
||||||
console.log("[FSAPIWatchAdapter] Consider using Chrome 124+ for real-time file watching");
|
console.log("[FSAPIWatchAdapter] Consider using Chrome 124+ for real-time file watching");
|
||||||
return Promise.resolve();
|
return Promise.resolve();
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const FileSystemObserver = (window as any).FileSystemObserver;
|
const FileSystemObserver = (compatGlobal as any).FileSystemObserver;
|
||||||
|
|
||||||
this.observer = new FileSystemObserver(async (records: any[]) => {
|
this.observer = new FileSystemObserver(async (records: any[]) => {
|
||||||
for (const record of records) {
|
for (const record of records) {
|
||||||
@@ -181,7 +182,7 @@ class FSAPIWatchAdapter implements IStorageEventWatchAdapter {
|
|||||||
if (changedHandle && changedHandle.kind === "file") {
|
if (changedHandle && changedHandle.kind === "file") {
|
||||||
const file = await changedHandle.getFile();
|
const file = await changedHandle.getFile();
|
||||||
const fileInfo = {
|
const fileInfo = {
|
||||||
path: relativePath as any,
|
path: relativePath,
|
||||||
stat: {
|
stat: {
|
||||||
size: file.size,
|
size: file.size,
|
||||||
mtime: file.lastModified,
|
mtime: file.lastModified,
|
||||||
@@ -199,7 +200,7 @@ class FSAPIWatchAdapter implements IStorageEventWatchAdapter {
|
|||||||
}
|
}
|
||||||
} else if (type === "disappeared") {
|
} else if (type === "disappeared") {
|
||||||
const fileInfo = {
|
const fileInfo = {
|
||||||
path: relativePath as any,
|
path: relativePath,
|
||||||
stat: {
|
stat: {
|
||||||
size: 0,
|
size: 0,
|
||||||
mtime: Date.now(),
|
mtime: Date.now(),
|
||||||
@@ -216,7 +217,7 @@ class FSAPIWatchAdapter implements IStorageEventWatchAdapter {
|
|||||||
if (changedHandle && changedHandle.kind === "file") {
|
if (changedHandle && changedHandle.kind === "file") {
|
||||||
const file = await changedHandle.getFile();
|
const file = await changedHandle.getFile();
|
||||||
const fileInfo = {
|
const fileInfo = {
|
||||||
path: relativePath as any,
|
path: relativePath,
|
||||||
stat: {
|
stat: {
|
||||||
size: file.size,
|
size: file.size,
|
||||||
mtime: file.lastModified,
|
mtime: file.lastModified,
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "livesync-webapp",
|
"name": "livesync-webapp",
|
||||||
"private": true,
|
"private": true,
|
||||||
"version": "0.0.1",
|
"version": "0.25.76-webapp",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"description": "Browser-based Self-hosted LiveSync using FileSystem API",
|
"description": "Browser-based Self-hosted LiveSync using FileSystem API",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
@@ -11,9 +11,16 @@
|
|||||||
"run:docker": "docker run -p 8002:80 livesync-webapp",
|
"run:docker": "docker run -p 8002:80 livesync-webapp",
|
||||||
"preview": "vite preview"
|
"preview": "vite preview"
|
||||||
},
|
},
|
||||||
"dependencies": {},
|
"dependencies": {
|
||||||
|
"octagonal-wheels": "^0.1.46"
|
||||||
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
"@playwright/test": "^1.58.2",
|
||||||
|
"@sveltejs/vite-plugin-svelte": "^7.1.2",
|
||||||
|
"playwright": "^1.58.2",
|
||||||
|
"svelte": "5.56.3",
|
||||||
"typescript": "5.9.3",
|
"typescript": "5.9.3",
|
||||||
"vite": "^7.3.1"
|
"vite": "^8.0.16",
|
||||||
|
"vite-plugin-istanbul": "^9.0.1"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ import type { ServiceContext } from "@lib/services/base/ServiceBase";
|
|||||||
import { FileAccessFSAPI } from "./FileAccessFSAPI";
|
import { FileAccessFSAPI } from "./FileAccessFSAPI";
|
||||||
import { ServiceFileAccessFSAPI } from "./ServiceFileAccessImpl";
|
import { ServiceFileAccessFSAPI } from "./ServiceFileAccessImpl";
|
||||||
import { ServiceDatabaseFileAccessFSAPI } from "./DatabaseFileAccess";
|
import { ServiceDatabaseFileAccessFSAPI } from "./DatabaseFileAccess";
|
||||||
import { StorageEventManagerFSAPI } from "../managers/StorageEventManagerFSAPI";
|
import { StorageEventManagerFSAPI } from "@/apps/webapp/managers/StorageEventManagerFSAPI";
|
||||||
import type { ServiceModules } from "@lib/interfaces/ServiceModule";
|
import type { ServiceModules } from "@lib/interfaces/ServiceModule";
|
||||||
import { ServiceFileHandler } from "@/serviceModules/FileHandler";
|
import { ServiceFileHandler } from "@/serviceModules/FileHandler";
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { FileAccessBase, type FileAccessBaseDependencies } from "@lib/serviceModules/FileAccessBase";
|
import { FileAccessBase, type FileAccessBaseDependencies } from "@lib/serviceModules/FileAccessBase";
|
||||||
import { FSAPIFileSystemAdapter } from "../adapters/FSAPIFileSystemAdapter";
|
import { FSAPIFileSystemAdapter } from "@/apps/webapp/adapters/FSAPIFileSystemAdapter";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* FileSystem API-specific implementation of FileAccessBase
|
* FileSystem API-specific implementation of FileAccessBase
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { ServiceFileAccessBase, type StorageAccessBaseDependencies } from "@lib/serviceModules/ServiceFileAccessBase";
|
import { ServiceFileAccessBase, type StorageAccessBaseDependencies } from "@lib/serviceModules/ServiceFileAccessBase";
|
||||||
import { FSAPIFileSystemAdapter } from "../adapters/FSAPIFileSystemAdapter";
|
import { FSAPIFileSystemAdapter } from "@/apps/webapp/adapters/FSAPIFileSystemAdapter";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* FileSystem API-specific implementation of ServiceFileAccess
|
* FileSystem API-specific implementation of ServiceFileAccess
|
||||||
|
|||||||
@@ -12,6 +12,7 @@
|
|||||||
import { LiveSyncWebApp } from "./main";
|
import { LiveSyncWebApp } from "./main";
|
||||||
import type { ObsidianLiveSyncSettings } from "@lib/common/types";
|
import type { ObsidianLiveSyncSettings } from "@lib/common/types";
|
||||||
import type { FilePathWithPrefix } from "@lib/common/types";
|
import type { FilePathWithPrefix } from "@lib/common/types";
|
||||||
|
import { compatGlobal } from "@lib/common/coreEnvFunctions.ts";
|
||||||
|
|
||||||
// --------------------------------------------------------------------------
|
// --------------------------------------------------------------------------
|
||||||
// Internal state – one app instance per page / browser context
|
// Internal state – one app instance per page / browser context
|
||||||
@@ -41,7 +42,7 @@ async function waitForIdle(core: any, timeoutMs = 60_000): Promise<void> {
|
|||||||
(core.services?.fileProcessing?.processing?.value ?? 0) +
|
(core.services?.fileProcessing?.processing?.value ?? 0) +
|
||||||
(core.services?.replication?.storageApplyingCount?.value ?? 0);
|
(core.services?.replication?.storageApplyingCount?.value ?? 0);
|
||||||
if (q === 0) return;
|
if (q === 0) return;
|
||||||
await new Promise<void>((r) => setTimeout(r, 300));
|
await new Promise<void>((r) => compatGlobal.setTimeout(r, 300));
|
||||||
}
|
}
|
||||||
throw new Error(`waitForIdle timed out after ${timeoutMs} ms`);
|
throw new Error(`waitForIdle timed out after ${timeoutMs} ms`);
|
||||||
}
|
}
|
||||||
@@ -116,7 +117,7 @@ export interface LiveSyncTestAPI {
|
|||||||
const livesyncTest: LiveSyncTestAPI = {
|
const livesyncTest: LiveSyncTestAPI = {
|
||||||
async init(vaultName: string, settings: Partial<ObsidianLiveSyncSettings>): Promise<void> {
|
async init(vaultName: string, settings: Partial<ObsidianLiveSyncSettings>): Promise<void> {
|
||||||
// Clean up any stale OPFS data from previous runs.
|
// Clean up any stale OPFS data from previous runs.
|
||||||
const opfsRoot = await navigator.storage.getDirectory();
|
const opfsRoot = await compatGlobal.navigator.storage.getDirectory();
|
||||||
try {
|
try {
|
||||||
await opfsRoot.removeEntry(vaultName, { recursive: true });
|
await opfsRoot.removeEntry(vaultName, { recursive: true });
|
||||||
} catch {
|
} catch {
|
||||||
@@ -200,4 +201,4 @@ const livesyncTest: LiveSyncTestAPI = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
// Expose on window for Playwright page.evaluate() calls.
|
// Expose on window for Playwright page.evaluate() calls.
|
||||||
(window as any).livesyncTest = livesyncTest;
|
(compatGlobal as any).livesyncTest = livesyncTest;
|
||||||
|
|||||||
@@ -17,7 +17,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { test, expect, type BrowserContext, type Page, type TestInfo } from "@playwright/test";
|
import { test, expect, type BrowserContext, type Page, type TestInfo } from "@playwright/test";
|
||||||
import type { LiveSyncTestAPI } from "../test-entry";
|
import type { LiveSyncTestAPI } from "@/apps/webapp/test-entry";
|
||||||
import { mkdirSync, writeFileSync } from "node:fs";
|
import { mkdirSync, writeFileSync } from "node:fs";
|
||||||
import path from "node:path";
|
import path from "node:path";
|
||||||
import { fileURLToPath } from "node:url";
|
import { fileURLToPath } from "node:url";
|
||||||
|
|||||||
@@ -21,7 +21,7 @@
|
|||||||
"noFallthroughCasesInSwitch": true,
|
"noFallthroughCasesInSwitch": true,
|
||||||
|
|
||||||
/* Path mapping */
|
/* Path mapping */
|
||||||
"baseUrl": ".",
|
// "baseUrl": ".",
|
||||||
"paths": {
|
"paths": {
|
||||||
"@/*": ["../../*"],
|
"@/*": ["../../*"],
|
||||||
"@lib/*": ["../../lib/src/*"]
|
"@lib/*": ["../../lib/src/*"]
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
import { compatGlobal } from "@lib/common/coreEnvFunctions.ts";
|
||||||
|
|
||||||
const HANDLE_DB_NAME = "livesync-webapp-handles";
|
const HANDLE_DB_NAME = "livesync-webapp-handles";
|
||||||
const HANDLE_STORE_NAME = "handles";
|
const HANDLE_STORE_NAME = "handles";
|
||||||
const LAST_USED_KEY = "meta:lastUsedVaultId";
|
const LAST_USED_KEY = "meta:lastUsedVaultId";
|
||||||
@@ -89,7 +91,7 @@ export class VaultHistoryStore {
|
|||||||
|
|
||||||
async getVaultHistory(): Promise<VaultHistoryItem[]> {
|
async getVaultHistory(): Promise<VaultHistoryItem[]> {
|
||||||
return this.withStore("readonly", async (store) => {
|
return this.withStore("readonly", async (store) => {
|
||||||
const keys = (await this.requestAsPromise(store.getAllKeys())) as IDBValidKey[];
|
const keys = (await this.requestAsPromise(store.getAllKeys()));
|
||||||
const values = (await this.requestAsPromise(store.getAll())) as unknown[];
|
const values = (await this.requestAsPromise(store.getAll())) as unknown[];
|
||||||
const items: VaultHistoryItem[] = [];
|
const items: VaultHistoryItem[] = [];
|
||||||
for (let i = 0; i < keys.length; i++) {
|
for (let i = 0; i < keys.length; i++) {
|
||||||
@@ -170,7 +172,7 @@ export class VaultHistoryStore {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async pickNewVault(): Promise<FileSystemDirectoryHandle> {
|
async pickNewVault(): Promise<FileSystemDirectoryHandle> {
|
||||||
const picker = (window as any).showDirectoryPicker;
|
const picker = (compatGlobal as any).showDirectoryPicker;
|
||||||
if (typeof picker !== "function") {
|
if (typeof picker !== "function") {
|
||||||
throw new Error("FileSystem API showDirectoryPicker is not supported in this browser");
|
throw new Error("FileSystem API showDirectoryPicker is not supported in this browser");
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,13 +13,20 @@ This pseudo client actually receives the data from other devices, and sends if s
|
|||||||
|
|
||||||
## How to use it?
|
## How to use it?
|
||||||
|
|
||||||
We can build the application by running the following command:
|
We can build the application from the repository root by running the following command:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
$ deno task build
|
npm run build -w webpeer
|
||||||
```
|
```
|
||||||
|
|
||||||
Then, open the `dist/index.html` in the browser. It can be configured as the same as the Self-hosted LiveSync (Same components are used[^1]).
|
Or from the package directory:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd src/apps/webpeer
|
||||||
|
npm run build
|
||||||
|
```
|
||||||
|
|
||||||
|
Then, open `dist/index.html` in the browser. It can be configured in the same way as Self-hosted LiveSync (the same components are used[^1]).
|
||||||
|
|
||||||
## Some notes
|
## Some notes
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "webpeer",
|
"name": "webpeer",
|
||||||
"private": true,
|
"private": true,
|
||||||
"version": "0.0.0",
|
"version": "0.25.76-webpeer",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "vite",
|
"dev": "vite",
|
||||||
@@ -11,15 +11,17 @@
|
|||||||
"preview": "vite preview",
|
"preview": "vite preview",
|
||||||
"check": "svelte-check --tsconfig ./tsconfig.app.json && tsc -p tsconfig.node.json"
|
"check": "svelte-check --tsconfig ./tsconfig.app.json && tsc -p tsconfig.node.json"
|
||||||
},
|
},
|
||||||
"dependencies": {},
|
"dependencies": {
|
||||||
|
"octagonal-wheels": "^0.1.46"
|
||||||
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"eslint-plugin-svelte": "^3.15.0",
|
"eslint-plugin-svelte": "^3.19.0",
|
||||||
"@sveltejs/vite-plugin-svelte": "^6.2.4",
|
"@sveltejs/vite-plugin-svelte": "^7.1.2",
|
||||||
"@tsconfig/svelte": "^5.0.8",
|
"@tsconfig/svelte": "^5.0.8",
|
||||||
"svelte": "5.41.1",
|
"svelte": "5.56.3",
|
||||||
"svelte-check": "^443.3",
|
"svelte-check": "^4.6.0",
|
||||||
"typescript": "5.9.3",
|
"typescript": "5.9.3",
|
||||||
"vite": "^7.3.1"
|
"vite": "^8.0.16"
|
||||||
},
|
},
|
||||||
"imports": {
|
"imports": {
|
||||||
"../../src/worker/bgWorker.ts": "../../src/worker/bgWorker.mock.ts",
|
"../../src/worker/bgWorker.ts": "../../src/worker/bgWorker.mock.ts",
|
||||||
|
|||||||
@@ -17,7 +17,9 @@ import {
|
|||||||
type PeerStatus,
|
type PeerStatus,
|
||||||
type PluginShim,
|
type PluginShim,
|
||||||
} from "@lib/replication/trystero/P2PReplicatorPaneCommon";
|
} from "@lib/replication/trystero/P2PReplicatorPaneCommon";
|
||||||
import { P2PLogCollector, type P2PReplicatorBase, useP2PReplicator } from "@lib/replication/trystero/P2PReplicatorCore";
|
import { useP2PReplicator } from "@lib/replication/trystero/P2PReplicatorCore";
|
||||||
|
import { P2PLogCollector } from "@lib/replication/trystero/P2PLogCollector";
|
||||||
|
import type { P2PReplicatorBase } from "@lib/replication/trystero/P2PReplicatorBase.ts";
|
||||||
import type { SimpleStore } from "octagonal-wheels/databases/SimpleStoreBase";
|
import type { SimpleStore } from "octagonal-wheels/databases/SimpleStoreBase";
|
||||||
import { reactiveSource } from "octagonal-wheels/dataobject/reactive_v2";
|
import { reactiveSource } from "octagonal-wheels/dataobject/reactive_v2";
|
||||||
import { EVENT_SETTING_SAVED } from "@lib/events/coreEvents";
|
import { EVENT_SETTING_SAVED } from "@lib/events/coreEvents";
|
||||||
@@ -28,9 +30,10 @@ import { ServiceContext } from "@lib/services/base/ServiceBase";
|
|||||||
import type { InjectableServiceHub } from "@lib/services/InjectableServices";
|
import type { InjectableServiceHub } from "@lib/services/InjectableServices";
|
||||||
import { Menu } from "@lib/services/implements/browser/Menu";
|
import { Menu } from "@lib/services/implements/browser/Menu";
|
||||||
import { SimpleStoreIDBv2 } from "octagonal-wheels/databases/SimpleStoreIDBv2";
|
import { SimpleStoreIDBv2 } from "octagonal-wheels/databases/SimpleStoreIDBv2";
|
||||||
import type { BrowserAPIService } from "@/lib/src/services/implements/browser/BrowserAPIService";
|
import type { BrowserAPIService } from "@lib/services/implements/browser/BrowserAPIService";
|
||||||
import type { InjectableSettingService } from "@/lib/src/services/implements/injectable/InjectableSettingService";
|
import type { InjectableSettingService } from "@lib/services/implements/injectable/InjectableSettingService";
|
||||||
import { LiveSyncTrysteroReplicator } from "@lib/replication/trystero/LiveSyncTrysteroReplicator";
|
import { LiveSyncTrysteroReplicator } from "@lib/replication/trystero/LiveSyncTrysteroReplicator";
|
||||||
|
import { compatGlobal } from "@lib/common/coreEnvFunctions.ts";
|
||||||
|
|
||||||
function addToList(item: string, list: string) {
|
function addToList(item: string, list: string) {
|
||||||
return unique(
|
return unique(
|
||||||
@@ -137,7 +140,7 @@ export class P2PReplicatorShim implements P2PReplicatorBase {
|
|||||||
|
|
||||||
this._initP2PReplicator();
|
this._initP2PReplicator();
|
||||||
|
|
||||||
setTimeout(() => {
|
compatGlobal.setTimeout(() => {
|
||||||
if (this.settings.P2P_AutoStart && this.settings.P2P_Enabled) {
|
if (this.settings.P2P_AutoStart && this.settings.P2P_Enabled) {
|
||||||
void this.open();
|
void this.open();
|
||||||
}
|
}
|
||||||
@@ -164,12 +167,12 @@ export class P2PReplicatorShim implements P2PReplicatorBase {
|
|||||||
getConfig(key: string) {
|
getConfig(key: string) {
|
||||||
const vaultName = this.services.vault.getVaultName();
|
const vaultName = this.services.vault.getVaultName();
|
||||||
const dbKey = `${vaultName}-${key}`;
|
const dbKey = `${vaultName}-${key}`;
|
||||||
return localStorage.getItem(dbKey);
|
return compatGlobal.localStorage.getItem(dbKey);
|
||||||
}
|
}
|
||||||
setConfig(key: string, value: string) {
|
setConfig(key: string, value: string) {
|
||||||
const vaultName = this.services.vault.getVaultName();
|
const vaultName = this.services.vault.getVaultName();
|
||||||
const dbKey = `${vaultName}-${key}`;
|
const dbKey = `${vaultName}-${key}`;
|
||||||
localStorage.setItem(dbKey, value);
|
compatGlobal.localStorage.setItem(dbKey, value);
|
||||||
}
|
}
|
||||||
|
|
||||||
getDeviceName(): string {
|
getDeviceName(): string {
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { Menu } from "@/lib/src/services/implements/browser/Menu";
|
import { Menu } from "@lib/services/implements/browser/Menu";
|
||||||
import { getDialogContext } from "@lib/services/implements/base/SvelteDialog";
|
import { getDialogContext } from "@lib/services/implements/base/SvelteDialog";
|
||||||
let result = $state<string | boolean>("");
|
let result = $state<string | boolean>("");
|
||||||
|
|
||||||
|
|||||||
@@ -1,9 +1,10 @@
|
|||||||
import { mount } from "svelte";
|
import { mount } from "svelte";
|
||||||
import "./app.css";
|
import "./app.css";
|
||||||
import App from "./App.svelte";
|
import App from "./App.svelte";
|
||||||
|
import { _activeDocument } from "@lib/common/coreEnvFunctions.ts";
|
||||||
|
|
||||||
const app = mount(App, {
|
const app = mount(App, {
|
||||||
target: document.getElementById("app")!,
|
target: _activeDocument.getElementById("app")!,
|
||||||
});
|
});
|
||||||
|
|
||||||
export default app;
|
export default app;
|
||||||
|
|||||||
@@ -1,9 +1,10 @@
|
|||||||
import { mount } from "svelte";
|
import { mount } from "svelte";
|
||||||
import "./app.css";
|
import "./app.css";
|
||||||
import App from "./UITest.svelte";
|
import App from "./UITest.svelte";
|
||||||
|
import { _activeDocument } from "@lib/common/coreEnvFunctions.ts";
|
||||||
|
|
||||||
const app = mount(App, {
|
const app = mount(App, {
|
||||||
target: document.getElementById("app")!,
|
target: _activeDocument.getElementById("app")!,
|
||||||
});
|
});
|
||||||
|
|
||||||
export default app;
|
export default app;
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"extends": "@tsconfig/svelte/tsconfig.json",
|
"extends": "../../../tsconfig.json",
|
||||||
"compilerOptions": {
|
"compilerOptions": {
|
||||||
"sourceRoot": "../",
|
// "sourceRoot": "../",
|
||||||
"target": "ESNext",
|
"target": "ESNext",
|
||||||
"useDefineForClassFields": true,
|
"useDefineForClassFields": true,
|
||||||
"module": "ESNext",
|
"module": "ESNext",
|
||||||
@@ -15,11 +15,13 @@
|
|||||||
"allowJs": true,
|
"allowJs": true,
|
||||||
"checkJs": true,
|
"checkJs": true,
|
||||||
"isolatedModules": true,
|
"isolatedModules": true,
|
||||||
|
"allowImportingTsExtensions": true,
|
||||||
"moduleDetection": "force",
|
"moduleDetection": "force",
|
||||||
"paths": {
|
"paths": {
|
||||||
"@/*": ["../../*"],
|
"@/*": ["../../*"],
|
||||||
"@lib/*": ["../../lib/src/*"]
|
"@lib/*": ["../../lib/src/*"]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"include": ["src/**/*.ts", "src/**/*.js", "src/**/*.svelte"]
|
"include": ["src/**/*.ts", "src/**/*.js", "src/**/*.svelte"],
|
||||||
|
"exclude": ["node_modules", "dist"]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { deleteDB, type IDBPDatabase, openDB } from "idb";
|
import { deleteDB, type IDBPDatabase, openDB } from "idb";
|
||||||
import type { KeyValueDatabase } from "../lib/src/interfaces/KeyValueDatabase.ts";
|
import type { KeyValueDatabase } from "@lib/interfaces/KeyValueDatabase.ts";
|
||||||
import { serialized } from "octagonal-wheels/concurrency/lock";
|
import { serialized } from "octagonal-wheels/concurrency/lock";
|
||||||
import { Logger } from "octagonal-wheels/common/logger";
|
import { Logger } from "octagonal-wheels/common/logger";
|
||||||
const databaseCache: { [key: string]: IDBPDatabase<any> } = {};
|
const databaseCache: { [key: string]: IDBPDatabase<any> } = {};
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { LOG_LEVEL_VERBOSE, Logger } from "@/lib/src/common/logger";
|
import { LOG_LEVEL_VERBOSE, Logger } from "@lib/common/logger";
|
||||||
import type { KeyValueDatabase } from "@/lib/src/interfaces/KeyValueDatabase";
|
import type { KeyValueDatabase } from "@lib/interfaces/KeyValueDatabase";
|
||||||
import { deleteDB, openDB, type IDBPDatabase } from "idb";
|
import { deleteDB, openDB, type IDBPDatabase } from "idb";
|
||||||
import { serialized } from "octagonal-wheels/concurrency/lock";
|
import { serialized } from "octagonal-wheels/concurrency/lock";
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { eventHub } from "../lib/src/hub/hub";
|
import { eventHub } from "@lib/hub/hub";
|
||||||
// import type ObsidianLiveSyncPlugin from "../main";
|
// import type ObsidianLiveSyncPlugin from "../main";
|
||||||
|
|
||||||
export const EVENT_PLUGIN_LOADED = "plugin-loaded";
|
export const EVENT_PLUGIN_LOADED = "plugin-loaded";
|
||||||
@@ -43,5 +43,5 @@ declare global {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export * from "../lib/src/events/coreEvents.ts";
|
export * from "@lib/events/coreEvents.ts";
|
||||||
export { eventHub };
|
export { eventHub };
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import type { TFile } from "../deps";
|
import type { TFile } from "@/deps";
|
||||||
import type { FilePathWithPrefix, LoadedEntry } from "../lib/src/common/types";
|
import type { FilePathWithPrefix, LoadedEntry } from "@lib/common/types";
|
||||||
|
|
||||||
export const EVENT_REQUEST_SHOW_HISTORY = "show-history";
|
export const EVENT_REQUEST_SHOW_HISTORY = "show-history";
|
||||||
|
|
||||||
|
|||||||
@@ -36,7 +36,7 @@ export async function generateReport(settings: ObsidianLiveSyncSettings, core: L
|
|||||||
const r = await requestToCouchDBWithCredentials(
|
const r = await requestToCouchDBWithCredentials(
|
||||||
settings.couchDB_URI,
|
settings.couchDB_URI,
|
||||||
credential,
|
credential,
|
||||||
window.origin,
|
compatGlobal.origin,
|
||||||
undefined,
|
undefined,
|
||||||
undefined,
|
undefined,
|
||||||
undefined,
|
undefined,
|
||||||
|
|||||||
+5
-5
@@ -1,6 +1,6 @@
|
|||||||
import { type PluginManifest, TFile } from "../deps.ts";
|
import { type PluginManifest, TFile } from "@/deps.ts";
|
||||||
import { type DatabaseEntry, type EntryBody, type FilePath } from "../lib/src/common/types.ts";
|
import { type DatabaseEntry, type EntryBody, type FilePath } from "@lib/common/types.ts";
|
||||||
export type { CacheData, FileEventItem } from "../lib/src/common/types.ts";
|
export type { CacheData, FileEventItem } from "@lib/common/types.ts";
|
||||||
|
|
||||||
export interface PluginDataEntry extends DatabaseEntry {
|
export interface PluginDataEntry extends DatabaseEntry {
|
||||||
deviceVaultName: string;
|
deviceVaultName: string;
|
||||||
@@ -51,7 +51,7 @@ export type queueItem = {
|
|||||||
|
|
||||||
export const FileWatchEventQueueMax = 10;
|
export const FileWatchEventQueueMax = 10;
|
||||||
|
|
||||||
export { configURIBase, configURIBaseQR } from "../lib/src/common/types.ts";
|
export { configURIBase, configURIBaseQR } from "@lib/common/types.ts";
|
||||||
|
|
||||||
export {
|
export {
|
||||||
CHeader,
|
CHeader,
|
||||||
@@ -61,4 +61,4 @@ export {
|
|||||||
ICHeaderEnd,
|
ICHeaderEnd,
|
||||||
ICHeaderLength,
|
ICHeaderLength,
|
||||||
ICXHeader,
|
ICXHeader,
|
||||||
} from "../lib/src/common/models/fileaccess.const.ts";
|
} from "@lib/common/models/fileaccess.const.ts";
|
||||||
|
|||||||
+9
-9
@@ -1,4 +1,4 @@
|
|||||||
import { normalizePath, Platform, TAbstractFile, type RequestUrlParam, requestUrl } from "../deps.ts";
|
import { normalizePath, Platform, TAbstractFile, type RequestUrlParam, requestUrl } from "@/deps.ts";
|
||||||
import {
|
import {
|
||||||
path2id_base,
|
path2id_base,
|
||||||
id2path_base,
|
id2path_base,
|
||||||
@@ -7,9 +7,9 @@ import {
|
|||||||
isValidFilenameInWidows,
|
isValidFilenameInWidows,
|
||||||
isValidFilenameInAndroid,
|
isValidFilenameInAndroid,
|
||||||
stripAllPrefixes,
|
stripAllPrefixes,
|
||||||
} from "../lib/src/string_and_binary/path.ts";
|
} from "@lib/string_and_binary/path.ts";
|
||||||
|
|
||||||
import { Logger } from "../lib/src/common/logger.ts";
|
import { Logger } from "@lib/common/logger.ts";
|
||||||
import {
|
import {
|
||||||
LOG_LEVEL_INFO,
|
LOG_LEVEL_INFO,
|
||||||
LOG_LEVEL_NOTICE,
|
LOG_LEVEL_NOTICE,
|
||||||
@@ -22,14 +22,14 @@ import {
|
|||||||
type FilePathWithPrefix,
|
type FilePathWithPrefix,
|
||||||
type UXFileInfo,
|
type UXFileInfo,
|
||||||
type UXFileInfoStub,
|
type UXFileInfoStub,
|
||||||
} from "../lib/src/common/types.ts";
|
} from "@lib/common/types.ts";
|
||||||
export { ICHeader, ICXHeader } from "./types.ts";
|
export { ICHeader, ICXHeader } from "./types.ts";
|
||||||
import { writeString } from "../lib/src/string_and_binary/convert.ts";
|
import { writeString } from "@lib/string_and_binary/convert.ts";
|
||||||
import { sameChangePairs } from "./stores.ts";
|
import { sameChangePairs } from "./stores.ts";
|
||||||
|
|
||||||
import { scheduleTask } from "octagonal-wheels/concurrency/task";
|
import { scheduleTask } from "octagonal-wheels/concurrency/task";
|
||||||
import { AuthorizationHeaderGenerator } from "../lib/src/replication/httplib.ts";
|
import { AuthorizationHeaderGenerator } from "@lib/replication/httplib.ts";
|
||||||
import type { KeyValueDatabase } from "../lib/src/interfaces/KeyValueDatabase.ts";
|
import type { KeyValueDatabase } from "@lib/interfaces/KeyValueDatabase.ts";
|
||||||
|
|
||||||
export { scheduleTask, cancelTask, cancelAllTasks } from "octagonal-wheels/concurrency/task";
|
export { scheduleTask, cancelTask, cancelAllTasks } from "octagonal-wheels/concurrency/task";
|
||||||
|
|
||||||
@@ -132,7 +132,7 @@ export const _requestToCouchDBFetch = async (
|
|||||||
method?: string
|
method?: string
|
||||||
) => {
|
) => {
|
||||||
const utf8str = String.fromCharCode.apply(null, [...writeString(`${username}:${password}`)]);
|
const utf8str = String.fromCharCode.apply(null, [...writeString(`${username}:${password}`)]);
|
||||||
const encoded = window.btoa(utf8str);
|
const encoded = compatGlobal.btoa(utf8str);
|
||||||
const authHeader = "Basic " + encoded;
|
const authHeader = "Basic " + encoded;
|
||||||
const transformedHeaders: Record<string, string> = {
|
const transformedHeaders: Record<string, string> = {
|
||||||
authorization: authHeader,
|
authorization: authHeader,
|
||||||
@@ -214,7 +214,7 @@ import { BASE_IS_NEW, EVEN, TARGET_IS_NEW } from "@lib/common/models/shared.cons
|
|||||||
export { BASE_IS_NEW, EVEN, TARGET_IS_NEW };
|
export { BASE_IS_NEW, EVEN, TARGET_IS_NEW };
|
||||||
// Why 2000? : ZIP FILE Does not have enough resolution.
|
// Why 2000? : ZIP FILE Does not have enough resolution.
|
||||||
import { compareMTime } from "@lib/common/utils.ts";
|
import { compareMTime } from "@lib/common/utils.ts";
|
||||||
import { _fetch } from "@/lib/src/common/coreEnvFunctions.ts";
|
import { _fetch, compatGlobal } from "@lib/common/coreEnvFunctions.ts";
|
||||||
export { compareMTime };
|
export { compareMTime };
|
||||||
function getKey(file: AnyEntry | string | UXFileInfoStub) {
|
function getKey(file: AnyEntry | string | UXFileInfoStub) {
|
||||||
const key = typeof file == "string" ? file : stripAllPrefixes(file.path);
|
const key = typeof file == "string" ? file : stripAllPrefixes(file.path);
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ import {
|
|||||||
diff_match_patch,
|
diff_match_patch,
|
||||||
Platform,
|
Platform,
|
||||||
addIcon,
|
addIcon,
|
||||||
} from "../../deps.ts";
|
} from "@/deps.ts";
|
||||||
|
|
||||||
import type {
|
import type {
|
||||||
EntryDoc,
|
EntryDoc,
|
||||||
@@ -19,7 +19,7 @@ import type {
|
|||||||
AnyEntry,
|
AnyEntry,
|
||||||
SavingEntry,
|
SavingEntry,
|
||||||
diff_result,
|
diff_result,
|
||||||
} from "../../lib/src/common/types.ts";
|
} from "@lib/common/types.ts";
|
||||||
import {
|
import {
|
||||||
CANCELLED,
|
CANCELLED,
|
||||||
LEAVE_TO_SUBSEQUENT,
|
LEAVE_TO_SUBSEQUENT,
|
||||||
@@ -29,8 +29,8 @@ import {
|
|||||||
LOG_LEVEL_VERBOSE,
|
LOG_LEVEL_VERBOSE,
|
||||||
MODE_SELECTIVE,
|
MODE_SELECTIVE,
|
||||||
MODE_SHINY,
|
MODE_SHINY,
|
||||||
} from "../../lib/src/common/types.ts";
|
} from "@lib/common/types.ts";
|
||||||
import { ICXHeader, PERIODIC_PLUGIN_SWEEP } from "../../common/types.ts";
|
import { ICXHeader, PERIODIC_PLUGIN_SWEEP } from "@/common/types.ts";
|
||||||
import {
|
import {
|
||||||
createBlob,
|
createBlob,
|
||||||
createSavingEntryFromLoadedEntry,
|
createSavingEntryFromLoadedEntry,
|
||||||
@@ -42,12 +42,12 @@ import {
|
|||||||
isDocContentSame,
|
isDocContentSame,
|
||||||
isLoadedEntry,
|
isLoadedEntry,
|
||||||
isObjectDifferent,
|
isObjectDifferent,
|
||||||
} from "../../lib/src/common/utils.ts";
|
} from "@lib/common/utils.ts";
|
||||||
import { digestHash } from "../../lib/src/string_and_binary/hash.ts";
|
import { digestHash } from "@lib/string_and_binary/hash.ts";
|
||||||
import { arrayBufferToBase64, decodeBinary, readString } from "../../lib/src/string_and_binary/convert.ts";
|
import { arrayBufferToBase64, decodeBinary, readString } from "@lib/string_and_binary/convert.ts";
|
||||||
import { serialized, shareRunningResult } from "octagonal-wheels/concurrency/lock";
|
import { serialized, shareRunningResult } from "octagonal-wheels/concurrency/lock";
|
||||||
import { LiveSyncCommands } from "../LiveSyncCommands.ts";
|
import { LiveSyncCommands } from "@/features/LiveSyncCommands.ts";
|
||||||
import { stripAllPrefixes } from "../../lib/src/string_and_binary/path.ts";
|
import { stripAllPrefixes } from "@lib/string_and_binary/path.ts";
|
||||||
import {
|
import {
|
||||||
EVEN,
|
EVEN,
|
||||||
disposeMemoObject,
|
disposeMemoObject,
|
||||||
@@ -57,20 +57,20 @@ import {
|
|||||||
memoObject,
|
memoObject,
|
||||||
retrieveMemoObject,
|
retrieveMemoObject,
|
||||||
scheduleTask,
|
scheduleTask,
|
||||||
} from "../../common/utils.ts";
|
} from "@/common/utils.ts";
|
||||||
import { PeriodicProcessor } from "@/common/PeriodicProcessor.ts";
|
import { PeriodicProcessor } from "@/common/PeriodicProcessor.ts";
|
||||||
import { JsonResolveModal } from "../HiddenFileCommon/JsonResolveModal.ts";
|
import { JsonResolveModal } from "@/features/HiddenFileCommon/JsonResolveModal.ts";
|
||||||
import { QueueProcessor } from "octagonal-wheels/concurrency/processor";
|
import { QueueProcessor } from "octagonal-wheels/concurrency/processor";
|
||||||
import { pluginScanningCount } from "../../lib/src/mock_and_interop/stores.ts";
|
import { pluginScanningCount } from "@lib/mock_and_interop/stores.ts";
|
||||||
import type ObsidianLiveSyncPlugin from "../../main.ts";
|
import type ObsidianLiveSyncPlugin from "@/main.ts";
|
||||||
import { base64ToArrayBuffer, base64ToString } from "octagonal-wheels/binary/base64";
|
import { base64ToArrayBuffer, base64ToString } from "octagonal-wheels/binary/base64";
|
||||||
import { ConflictResolveModal } from "../../modules/features/InteractiveConflictResolving/ConflictResolveModal.ts";
|
import { ConflictResolveModal } from "@/modules/features/InteractiveConflictResolving/ConflictResolveModal.ts";
|
||||||
import { Semaphore } from "octagonal-wheels/concurrency/semaphore";
|
import { Semaphore } from "octagonal-wheels/concurrency/semaphore";
|
||||||
import { EVENT_REQUEST_OPEN_PLUGIN_SYNC_DIALOG, eventHub } from "../../common/events.ts";
|
import { EVENT_REQUEST_OPEN_PLUGIN_SYNC_DIALOG, eventHub } from "@/common/events.ts";
|
||||||
import { PluginDialogModal } from "./PluginDialogModal.ts";
|
import { PluginDialogModal } from "./PluginDialogModal.ts";
|
||||||
import { $msg } from "@/lib/src/common/i18n.ts";
|
import { $msg } from "@lib/common/i18n.ts";
|
||||||
import type { InjectableServiceHub } from "../../lib/src/services/InjectableServices.ts";
|
import type { InjectableServiceHub } from "@lib/services/InjectableServices.ts";
|
||||||
import type { LiveSyncCore } from "../../main.ts";
|
import type { LiveSyncCore } from "@/main.ts";
|
||||||
import { LiveSyncError } from "@lib/common/LSError.ts";
|
import { LiveSyncError } from "@lib/common/LSError.ts";
|
||||||
|
|
||||||
const d = "\u200b";
|
const d = "\u200b";
|
||||||
|
|||||||
@@ -5,10 +5,10 @@
|
|||||||
type IPluginDataExDisplay,
|
type IPluginDataExDisplay,
|
||||||
type PluginDataExFile,
|
type PluginDataExFile,
|
||||||
} from "./CmdConfigSync.ts";
|
} from "./CmdConfigSync.ts";
|
||||||
import { Logger } from "../../lib/src/common/logger";
|
import { Logger } from "@lib/common/logger";
|
||||||
import { type FilePath, LOG_LEVEL_INFO, LOG_LEVEL_NOTICE, LOG_LEVEL_VERBOSE } from "../../lib/src/common/types";
|
import { type FilePath, LOG_LEVEL_INFO, LOG_LEVEL_NOTICE, LOG_LEVEL_VERBOSE } from "@lib/common/types";
|
||||||
import { getDocData, timeDeltaToHumanReadable, unique } from "../../lib/src/common/utils";
|
import { getDocData, timeDeltaToHumanReadable, unique } from "@lib/common/utils";
|
||||||
import type ObsidianLiveSyncPlugin from "../../main";
|
import type ObsidianLiveSyncPlugin from "@/main";
|
||||||
// import { askString } from "../../common/utils";
|
// import { askString } from "../../common/utils";
|
||||||
import { Menu } from "@/deps.ts";
|
import { Menu } from "@/deps.ts";
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { mount, unmount } from "svelte";
|
import { mount, unmount } from "svelte";
|
||||||
import { App, Modal } from "../../deps.ts";
|
import { App, Modal } from "@/deps.ts";
|
||||||
import ObsidianLiveSyncPlugin from "../../main.ts";
|
import ObsidianLiveSyncPlugin from "@/main.ts";
|
||||||
import PluginPane from "./PluginPane.svelte";
|
import PluginPane from "./PluginPane.svelte";
|
||||||
export class PluginDialogModal extends Modal {
|
export class PluginDialogModal extends Modal {
|
||||||
plugin: ObsidianLiveSyncPlugin;
|
plugin: ObsidianLiveSyncPlugin;
|
||||||
@@ -16,9 +16,11 @@ export class PluginDialogModal extends Modal {
|
|||||||
|
|
||||||
override onOpen() {
|
override onOpen() {
|
||||||
const { contentEl } = this;
|
const { contentEl } = this;
|
||||||
this.contentEl.style.overflow = "auto";
|
this.contentEl.setCssStyles({
|
||||||
this.contentEl.style.display = "flex";
|
overflow: "auto",
|
||||||
this.contentEl.style.flexDirection = "column";
|
display: "flex",
|
||||||
|
flexDirection: "column",
|
||||||
|
});
|
||||||
this.titleEl.setText("Customization Sync (Beta3)");
|
this.titleEl.setText("Customization Sync (Beta3)");
|
||||||
if (!this.component) {
|
if (!this.component) {
|
||||||
this.component = mount(PluginPane, {
|
this.component = mount(PluginPane, {
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { onMount } from "svelte";
|
import { onMount } from "svelte";
|
||||||
import ObsidianLiveSyncPlugin from "../../main";
|
import ObsidianLiveSyncPlugin from "@/main";
|
||||||
import {
|
import {
|
||||||
ConfigSync,
|
ConfigSync,
|
||||||
type IPluginDataExDisplay,
|
type IPluginDataExDisplay,
|
||||||
@@ -11,16 +11,16 @@
|
|||||||
} from "./CmdConfigSync.ts";
|
} from "./CmdConfigSync.ts";
|
||||||
import PluginCombo from "./PluginCombo.svelte";
|
import PluginCombo from "./PluginCombo.svelte";
|
||||||
import { Menu, type PluginManifest } from "@/deps.ts";
|
import { Menu, type PluginManifest } from "@/deps.ts";
|
||||||
import { unique } from "../../lib/src/common/utils";
|
import { unique } from "@lib/common/utils";
|
||||||
import {
|
import {
|
||||||
MODE_SELECTIVE,
|
MODE_SELECTIVE,
|
||||||
MODE_AUTOMATIC,
|
MODE_AUTOMATIC,
|
||||||
MODE_PAUSED,
|
MODE_PAUSED,
|
||||||
type SYNC_MODE,
|
type SYNC_MODE,
|
||||||
MODE_SHINY,
|
MODE_SHINY,
|
||||||
} from "../../lib/src/common/types";
|
} from "@lib/common/types";
|
||||||
import { normalizePath } from "../../deps";
|
import { normalizePath } from "@/deps";
|
||||||
import { HiddenFileSync } from "../HiddenFileSync/CmdHiddenFileSync.ts";
|
import { HiddenFileSync } from "@/features/HiddenFileSync/CmdHiddenFileSync.ts";
|
||||||
import { LOG_LEVEL_NOTICE, Logger } from "octagonal-wheels/common/logger";
|
import { LOG_LEVEL_NOTICE, Logger } from "octagonal-wheels/common/logger";
|
||||||
import type { LiveSyncBaseCore } from "@/LiveSyncBaseCore.ts";
|
import type { LiveSyncBaseCore } from "@/LiveSyncBaseCore.ts";
|
||||||
export let plugin: ObsidianLiveSyncPlugin;
|
export let plugin: ObsidianLiveSyncPlugin;
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { App, Modal } from "../../deps.ts";
|
import { App, Modal } from "@/deps.ts";
|
||||||
import { type FilePath, type LoadedEntry } from "../../lib/src/common/types.ts";
|
import { type FilePath, type LoadedEntry } from "@lib/common/types.ts";
|
||||||
import JsonResolvePane from "./JsonResolvePane.svelte";
|
import JsonResolvePane from "./JsonResolvePane.svelte";
|
||||||
import { waitForSignal } from "../../lib/src/common/utils.ts";
|
import { waitForSignal } from "@lib/common/utils.ts";
|
||||||
import { mount, unmount } from "svelte";
|
import { mount, unmount } from "svelte";
|
||||||
|
|
||||||
export class JsonResolveModal extends Modal {
|
export class JsonResolveModal extends Modal {
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { type Diff, DIFF_DELETE, DIFF_INSERT, diff_match_patch } from "../../deps.ts";
|
import { type Diff, DIFF_DELETE, DIFF_INSERT, diff_match_patch } from "@/deps.ts";
|
||||||
import type { FilePath, LoadedEntry } from "../../lib/src/common/types.ts";
|
import type { FilePath, LoadedEntry } from "@lib/common/types.ts";
|
||||||
import { decodeBinary, readString } from "../../lib/src/string_and_binary/convert.ts";
|
import { decodeBinary, readString } from "@lib/string_and_binary/convert.ts";
|
||||||
import { getDocData, isObjectDifferent, mergeObject } from "../../lib/src/common/utils.ts";
|
import { getDocData, isObjectDifferent, mergeObject } from "@lib/common/utils.ts";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
docs?: LoadedEntry[];
|
docs?: LoadedEntry[];
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { type PluginManifest, type ListedFiles } from "../../deps.ts";
|
import { type PluginManifest, type ListedFiles } from "@/deps.ts";
|
||||||
import {
|
import {
|
||||||
type LoadedEntry,
|
type LoadedEntry,
|
||||||
type FilePathWithPrefix,
|
type FilePathWithPrefix,
|
||||||
@@ -15,8 +15,8 @@ import {
|
|||||||
LOG_LEVEL_DEBUG,
|
LOG_LEVEL_DEBUG,
|
||||||
type MetaEntry,
|
type MetaEntry,
|
||||||
type UXDataWriteOptions,
|
type UXDataWriteOptions,
|
||||||
} from "../../lib/src/common/types.ts";
|
} from "@lib/common/types.ts";
|
||||||
import { type InternalFileInfo, ICHeader, ICHeaderEnd } from "../../common/types.ts";
|
import { type InternalFileInfo, ICHeader, ICHeaderEnd } from "@/common/types.ts";
|
||||||
import {
|
import {
|
||||||
readAsBlob,
|
readAsBlob,
|
||||||
isDocContentSame,
|
isDocContentSame,
|
||||||
@@ -26,7 +26,7 @@ import {
|
|||||||
fireAndForget,
|
fireAndForget,
|
||||||
type CustomRegExp,
|
type CustomRegExp,
|
||||||
getFileRegExp,
|
getFileRegExp,
|
||||||
} from "../../lib/src/common/utils.ts";
|
} from "@lib/common/utils.ts";
|
||||||
import {
|
import {
|
||||||
compareMTime,
|
compareMTime,
|
||||||
isInternalMetadata,
|
isInternalMetadata,
|
||||||
@@ -39,17 +39,17 @@ import {
|
|||||||
BASE_IS_NEW,
|
BASE_IS_NEW,
|
||||||
EVEN,
|
EVEN,
|
||||||
displayRev,
|
displayRev,
|
||||||
} from "../../common/utils.ts";
|
} from "@/common/utils.ts";
|
||||||
import { PeriodicProcessor } from "@/common/PeriodicProcessor.ts";
|
import { PeriodicProcessor } from "@/common/PeriodicProcessor.ts";
|
||||||
import { serialized, skipIfDuplicated } from "octagonal-wheels/concurrency/lock";
|
import { serialized, skipIfDuplicated } from "octagonal-wheels/concurrency/lock";
|
||||||
import { JsonResolveModal } from "../HiddenFileCommon/JsonResolveModal.ts";
|
import { JsonResolveModal } from "@/features/HiddenFileCommon/JsonResolveModal.ts";
|
||||||
import { LiveSyncCommands } from "../LiveSyncCommands.ts";
|
import { LiveSyncCommands } from "@/features/LiveSyncCommands.ts";
|
||||||
import { addPrefix, stripAllPrefixes } from "../../lib/src/string_and_binary/path.ts";
|
import { addPrefix, stripAllPrefixes } from "@lib/string_and_binary/path.ts";
|
||||||
import { QueueProcessor } from "octagonal-wheels/concurrency/processor";
|
import { QueueProcessor } from "octagonal-wheels/concurrency/processor";
|
||||||
import { hiddenFilesEventCount, hiddenFilesProcessingCount } from "../../lib/src/mock_and_interop/stores.ts";
|
import { hiddenFilesEventCount, hiddenFilesProcessingCount } from "@lib/mock_and_interop/stores.ts";
|
||||||
import { EVENT_SETTING_SAVED, eventHub } from "../../common/events.ts";
|
import { EVENT_SETTING_SAVED, eventHub } from "@/common/events.ts";
|
||||||
import { Semaphore } from "octagonal-wheels/concurrency/semaphore";
|
import { Semaphore } from "octagonal-wheels/concurrency/semaphore";
|
||||||
import type { LiveSyncCore } from "../../main.ts";
|
import type { LiveSyncCore } from "@/main.ts";
|
||||||
import { tryGetFilePath } from "@lib/common/utils.doc.ts";
|
import { tryGetFilePath } from "@lib/common/utils.doc.ts";
|
||||||
type SyncDirection = "push" | "pull" | "safe" | "pullForce" | "pushForce";
|
type SyncDirection = "push" | "pull" | "safe" | "pullForce" | "pushForce";
|
||||||
|
|
||||||
|
|||||||
@@ -7,12 +7,12 @@ import {
|
|||||||
type FilePath,
|
type FilePath,
|
||||||
type FilePathWithPrefix,
|
type FilePathWithPrefix,
|
||||||
type LOG_LEVEL,
|
type LOG_LEVEL,
|
||||||
} from "../lib/src/common/types.ts";
|
} from "@lib/common/types.ts";
|
||||||
import type ObsidianLiveSyncPlugin from "../main.ts";
|
import type ObsidianLiveSyncPlugin from "@/main.ts";
|
||||||
import { MARK_DONE } from "../modules/features/ModuleLog.ts";
|
import { MARK_DONE } from "@/modules/features/ModuleLog.ts";
|
||||||
import type { LiveSyncCore } from "../main.ts";
|
import type { LiveSyncCore } from "@/main.ts";
|
||||||
import { __$checkInstanceBinding } from "../lib/src/dev/checks.ts";
|
import { __$checkInstanceBinding } from "@lib/dev/checks.ts";
|
||||||
import { createInstanceLogFunction } from "@/lib/src/services/lib/logUtils.ts";
|
import { createInstanceLogFunction } from "@lib/services/lib/logUtils.ts";
|
||||||
|
|
||||||
let noticeIndex = 0;
|
let noticeIndex = 0;
|
||||||
export abstract class LiveSyncCommands {
|
export abstract class LiveSyncCommands {
|
||||||
|
|||||||
@@ -9,14 +9,14 @@ import {
|
|||||||
type EntryLeaf,
|
type EntryLeaf,
|
||||||
type FilePathWithPrefix,
|
type FilePathWithPrefix,
|
||||||
type MetaEntry,
|
type MetaEntry,
|
||||||
} from "../../lib/src/common/types";
|
} from "@lib/common/types";
|
||||||
import { getNoFromRev } from "../../lib/src/pouchdb/LiveSyncLocalDB";
|
import { getNoFromRev } from "@lib/pouchdb/LiveSyncLocalDB";
|
||||||
import { LiveSyncCommands } from "../LiveSyncCommands";
|
import { LiveSyncCommands } from "@/features/LiveSyncCommands";
|
||||||
import { serialized } from "octagonal-wheels/concurrency/lock_v2";
|
import { serialized } from "octagonal-wheels/concurrency/lock_v2";
|
||||||
import { arrayToChunkedArray } from "octagonal-wheels/collection";
|
import { arrayToChunkedArray } from "octagonal-wheels/collection";
|
||||||
import { EVENT_ANALYSE_DB_USAGE, EVENT_REQUEST_PERFORM_GC_V3, eventHub } from "@/common/events";
|
import { EVENT_ANALYSE_DB_USAGE, EVENT_REQUEST_PERFORM_GC_V3, eventHub } from "@/common/events";
|
||||||
import type { LiveSyncCouchDBReplicator } from "@/lib/src/replication/couchdb/LiveSyncReplicator";
|
import type { LiveSyncCouchDBReplicator } from "@lib/replication/couchdb/LiveSyncReplicator";
|
||||||
import { delay } from "@/lib/src/common/utils";
|
import { delay } from "@lib/common/utils";
|
||||||
// import { _requestToCouchDB } from "@/common/utils";
|
// import { _requestToCouchDB } from "@/common/utils";
|
||||||
const DB_KEY_SEQ = "gc-seq";
|
const DB_KEY_SEQ = "gc-seq";
|
||||||
const DB_KEY_CHUNK_SET = "chunk-set";
|
const DB_KEY_CHUNK_SET = "chunk-set";
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { App, Modal } from "@/deps.ts";
|
import { App, Modal } from "@/deps.ts";
|
||||||
import P2POpenReplicationPane from "./P2POpenReplicationPane.svelte";
|
import P2POpenReplicationPane from "./P2POpenReplicationPane.svelte";
|
||||||
import { mount, unmount } from "svelte";
|
import { mount, unmount } from "svelte";
|
||||||
import type { LiveSyncTrysteroReplicator } from "@/lib/src/replication/trystero/LiveSyncTrysteroReplicator";
|
import type { LiveSyncTrysteroReplicator } from "@lib/replication/trystero/LiveSyncTrysteroReplicator";
|
||||||
|
|
||||||
export type P2POpenReplicationModalCallback = {
|
export type P2POpenReplicationModalCallback = {
|
||||||
onSync: (peerId: string) => Promise<void>;
|
onSync: (peerId: string) => Promise<void>;
|
||||||
|
|||||||
@@ -9,7 +9,7 @@
|
|||||||
// import type { TrysteroReplicator } from "@lib/replication/trystero/TrysteroReplicator";
|
// import type { TrysteroReplicator } from "@lib/replication/trystero/TrysteroReplicator";
|
||||||
import { LOG_LEVEL_NOTICE, LOG_LEVEL_INFO } from "@lib/common/types";
|
import { LOG_LEVEL_NOTICE, LOG_LEVEL_INFO } from "@lib/common/types";
|
||||||
import { Logger } from "@lib/common/logger";
|
import { Logger } from "@lib/common/logger";
|
||||||
import type { LiveSyncTrysteroReplicator } from "@/lib/src/replication/trystero/LiveSyncTrysteroReplicator";
|
import type { LiveSyncTrysteroReplicator } from "@lib/replication/trystero/LiveSyncTrysteroReplicator";
|
||||||
import { delay, fireAndForget } from "octagonal-wheels/promises";
|
import { delay, fireAndForget } from "octagonal-wheels/promises";
|
||||||
import P2PServerStatusCard from "./P2PServerStatusCard.svelte";
|
import P2PServerStatusCard from "./P2PServerStatusCard.svelte";
|
||||||
|
|
||||||
|
|||||||
@@ -1,13 +1,13 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { onMount, setContext } from "svelte";
|
import { onMount, setContext } from "svelte";
|
||||||
import { AutoAccepting, DEFAULT_SETTINGS, type P2PSyncSetting } from "../../../lib/src/common/types";
|
import { AutoAccepting, DEFAULT_SETTINGS, type P2PSyncSetting } from "@lib/common/types";
|
||||||
import {
|
import {
|
||||||
AcceptedStatus,
|
AcceptedStatus,
|
||||||
ConnectionStatus,
|
ConnectionStatus,
|
||||||
type PeerStatus,
|
type PeerStatus,
|
||||||
} from "@lib/replication/trystero/P2PReplicatorPaneCommon";
|
} from "@lib/replication/trystero/P2PReplicatorPaneCommon";
|
||||||
import type { LiveSyncTrysteroReplicator } from "@lib/replication/trystero/LiveSyncTrysteroReplicator";
|
import type { LiveSyncTrysteroReplicator } from "@lib/replication/trystero/LiveSyncTrysteroReplicator";
|
||||||
import PeerStatusRow from "../P2PReplicator/PeerStatusRow.svelte";
|
import PeerStatusRow from "@/features/P2PSync/P2PReplicator/PeerStatusRow.svelte";
|
||||||
import { EVENT_LAYOUT_READY, eventHub } from "@/common/events";
|
import { EVENT_LAYOUT_READY, eventHub } from "@/common/events";
|
||||||
import {
|
import {
|
||||||
type PeerInfo,
|
type PeerInfo,
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ import { LOG_LEVEL_NOTICE, REMOTE_P2P } from "@lib/common/types.ts";
|
|||||||
import { Logger } from "@lib/common/logger.ts";
|
import { Logger } from "@lib/common/logger.ts";
|
||||||
import { EVENT_P2P_PEER_SHOW_EXTRA_MENU, type PeerStatus } from "@lib/replication/trystero/P2PReplicatorPaneCommon.ts";
|
import { EVENT_P2P_PEER_SHOW_EXTRA_MENU, type PeerStatus } from "@lib/replication/trystero/P2PReplicatorPaneCommon.ts";
|
||||||
import type { LiveSyncBaseCore } from "@/LiveSyncBaseCore.ts";
|
import type { LiveSyncBaseCore } from "@/LiveSyncBaseCore.ts";
|
||||||
import type { P2PPaneParams } from "@/lib/src/replication/trystero/UseP2PReplicatorResult";
|
import type { P2PPaneParams } from "@lib/replication/trystero/UseP2PReplicatorResult";
|
||||||
export const VIEW_TYPE_P2P = "p2p-replicator";
|
export const VIEW_TYPE_P2P = "p2p-replicator";
|
||||||
|
|
||||||
function addToList(item: string, list: string) {
|
function addToList(item: string, list: string) {
|
||||||
|
|||||||
@@ -9,9 +9,9 @@
|
|||||||
EVENT_P2P_REPLICATOR_STATUS,
|
EVENT_P2P_REPLICATOR_STATUS,
|
||||||
} from "@lib/replication/trystero/TrysteroReplicatorP2PServer";
|
} from "@lib/replication/trystero/TrysteroReplicatorP2PServer";
|
||||||
import { EVENT_SETTING_SAVED } from "@lib/events/coreEvents";
|
import { EVENT_SETTING_SAVED } from "@lib/events/coreEvents";
|
||||||
import type { LiveSyncTrysteroReplicator } from "@/lib/src/replication/trystero/LiveSyncTrysteroReplicator";
|
import type { LiveSyncTrysteroReplicator } from "@lib/replication/trystero/LiveSyncTrysteroReplicator";
|
||||||
import type { P2PReplicatorStatus } from "@/lib/src/replication/trystero/TrysteroReplicator";
|
import type { P2PReplicatorStatus } from "@lib/replication/trystero/TrysteroReplicator";
|
||||||
import { extractP2PRoomSuffix } from "@/lib/src/common/utils";
|
import { extractP2PRoomSuffix } from "@lib/common/utils";
|
||||||
import type { LiveSyncBaseCore } from "@/LiveSyncBaseCore";
|
import type { LiveSyncBaseCore } from "@/LiveSyncBaseCore";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ import { WorkspaceLeaf } from "@/deps.ts";
|
|||||||
import { mount } from "svelte";
|
import { mount } from "svelte";
|
||||||
import { SvelteItemView } from "@/common/SvelteItemView.ts";
|
import { SvelteItemView } from "@/common/SvelteItemView.ts";
|
||||||
import type { LiveSyncBaseCore } from "@/LiveSyncBaseCore.ts";
|
import type { LiveSyncBaseCore } from "@/LiveSyncBaseCore.ts";
|
||||||
import type { P2PPaneParams } from "@/lib/src/replication/trystero/UseP2PReplicatorResult";
|
import type { P2PPaneParams } from "@lib/replication/trystero/UseP2PReplicatorResult";
|
||||||
import P2PServerStatusPane from "./P2PServerStatusPane.svelte";
|
import P2PServerStatusPane from "./P2PServerStatusPane.svelte";
|
||||||
|
|
||||||
export const VIEW_TYPE_P2P_SERVER_STATUS = "p2p-server-status";
|
export const VIEW_TYPE_P2P_SERVER_STATUS = "p2p-server-status";
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { getContext } from "svelte";
|
import { getContext } from "svelte";
|
||||||
import { AcceptedStatus, type PeerStatus } from "../../../lib/src/replication/trystero/P2PReplicatorPaneCommon";
|
import { AcceptedStatus, type PeerStatus } from "@lib/replication/trystero/P2PReplicatorPaneCommon";
|
||||||
import type { LiveSyncTrysteroReplicator } from "../../../lib/src/replication/trystero/LiveSyncTrysteroReplicator";
|
import type { LiveSyncTrysteroReplicator } from "@lib/replication/trystero/LiveSyncTrysteroReplicator";
|
||||||
import { eventHub } from "../../../common/events";
|
import { eventHub } from "@/common/events";
|
||||||
import { EVENT_P2P_PEER_SHOW_EXTRA_MENU } from "../../../lib/src/replication/trystero/P2PReplicatorPaneCommon";
|
import { EVENT_P2P_PEER_SHOW_EXTRA_MENU } from "@lib/replication/trystero/P2PReplicatorPaneCommon";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
peerStatus: PeerStatus;
|
peerStatus: PeerStatus;
|
||||||
|
|||||||
+1
-1
Submodule src/lib updated: 5a552b3ec4...c926417f82
@@ -3,7 +3,7 @@ import type { AnyEntry, FilePathWithPrefix } from "@lib/common/types";
|
|||||||
import type { IMinimumLiveSyncCommands, LiveSyncBaseCore } from "@/LiveSyncBaseCore";
|
import type { IMinimumLiveSyncCommands, LiveSyncBaseCore } from "@/LiveSyncBaseCore";
|
||||||
import { stripAllPrefixes } from "@lib/string_and_binary/path";
|
import { stripAllPrefixes } from "@lib/string_and_binary/path";
|
||||||
import { createInstanceLogFunction } from "@lib/services/lib/logUtils";
|
import { createInstanceLogFunction } from "@lib/services/lib/logUtils";
|
||||||
import type { ServiceContext } from "@/lib/src/services/base/ServiceBase";
|
import type { ServiceContext } from "@lib/services/base/ServiceBase";
|
||||||
|
|
||||||
export abstract class AbstractModule<
|
export abstract class AbstractModule<
|
||||||
T extends LiveSyncBaseCore<ServiceContext, IMinimumLiveSyncCommands> = LiveSyncBaseCore<
|
T extends LiveSyncBaseCore<ServiceContext, IMinimumLiveSyncCommands> = LiveSyncBaseCore<
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { type Prettify } from "../lib/src/common/types";
|
import { type Prettify } from "@lib/common/types";
|
||||||
import type { LiveSyncCore } from "../main";
|
import type { LiveSyncCore } from "@/main";
|
||||||
import type ObsidianLiveSyncPlugin from "../main";
|
import type ObsidianLiveSyncPlugin from "@/main";
|
||||||
import { AbstractModule } from "./AbstractModule.ts";
|
import { AbstractModule } from "./AbstractModule.ts";
|
||||||
import type { ChainableExecuteFunction, OverridableFunctionsKeys } from "./ModuleTypes";
|
import type { ChainableExecuteFunction, OverridableFunctionsKeys } from "./ModuleTypes";
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import type { Prettify } from "../lib/src/common/types";
|
import type { Prettify } from "@lib/common/types";
|
||||||
import type { LiveSyncCore } from "../main";
|
import type { LiveSyncCore } from "@/main";
|
||||||
|
|
||||||
export type OverridableFunctionsKeys<T> = {
|
export type OverridableFunctionsKeys<T> = {
|
||||||
[K in keyof T as K extends `$${string}` ? K : never]: T[K];
|
[K in keyof T as K extends `$${string}` ? K : never]: T[K];
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { PeriodicProcessor } from "@/common/PeriodicProcessor";
|
import { PeriodicProcessor } from "@/common/PeriodicProcessor";
|
||||||
import type { LiveSyncCore } from "../../main";
|
import type { LiveSyncCore } from "@/main";
|
||||||
import { AbstractModule } from "../AbstractModule";
|
import { AbstractModule } from "@/modules/AbstractModule";
|
||||||
|
|
||||||
export class ModulePeriodicProcess extends AbstractModule {
|
export class ModulePeriodicProcess extends AbstractModule {
|
||||||
periodicSyncProcessor = new PeriodicProcessor(this.core, async () => await this.services.replication.replicate());
|
periodicSyncProcessor = new PeriodicProcessor(this.core, async () => await this.services.replication.replicate());
|
||||||
|
|||||||
@@ -1,18 +1,18 @@
|
|||||||
import type PouchDB from "pouchdb-core";
|
import type PouchDB from "pouchdb-core";
|
||||||
import { fireAndForget } from "octagonal-wheels/promises";
|
import { fireAndForget } from "octagonal-wheels/promises";
|
||||||
import { AbstractModule } from "../AbstractModule";
|
import { AbstractModule } from "@/modules/AbstractModule";
|
||||||
import { Logger, LOG_LEVEL_NOTICE, LOG_LEVEL_INFO } from "octagonal-wheels/common/logger";
|
import { Logger, LOG_LEVEL_NOTICE, LOG_LEVEL_INFO } from "octagonal-wheels/common/logger";
|
||||||
import { skipIfDuplicated } from "octagonal-wheels/concurrency/lock";
|
import { skipIfDuplicated } from "octagonal-wheels/concurrency/lock";
|
||||||
import { balanceChunkPurgedDBs } from "@lib/pouchdb/chunks";
|
import { balanceChunkPurgedDBs } from "@lib/pouchdb/chunks";
|
||||||
import { purgeUnreferencedChunks } from "@lib/pouchdb/chunks";
|
import { purgeUnreferencedChunks } from "@lib/pouchdb/chunks";
|
||||||
import { LiveSyncCouchDBReplicator } from "../../lib/src/replication/couchdb/LiveSyncReplicator";
|
import { LiveSyncCouchDBReplicator } from "@lib/replication/couchdb/LiveSyncReplicator";
|
||||||
import { type EntryDoc, type RemoteType } from "../../lib/src/common/types";
|
import { type EntryDoc, type RemoteType } from "@lib/common/types";
|
||||||
|
|
||||||
import { scheduleTask } from "octagonal-wheels/concurrency/task";
|
import { scheduleTask } from "octagonal-wheels/concurrency/task";
|
||||||
import { EVENT_FILE_SAVED, EVENT_SETTING_SAVED, eventHub } from "../../common/events";
|
import { EVENT_FILE_SAVED, EVENT_SETTING_SAVED, eventHub } from "@/common/events";
|
||||||
|
|
||||||
import { $msg } from "../../lib/src/common/i18n";
|
import { $msg } from "@lib/common/i18n";
|
||||||
import type { LiveSyncCore } from "../../main";
|
import type { LiveSyncCore } from "@/main";
|
||||||
import { ReplicateResultProcessor } from "./ReplicateResultProcessor";
|
import { ReplicateResultProcessor } from "./ReplicateResultProcessor";
|
||||||
import { UnresolvedErrorManager } from "@lib/services/base/UnresolvedErrorManager";
|
import { UnresolvedErrorManager } from "@lib/services/base/UnresolvedErrorManager";
|
||||||
import { clearHandlers } from "@lib/replication/SyncParamsHandler";
|
import { clearHandlers } from "@lib/replication/SyncParamsHandler";
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
import { fireAndForget } from "octagonal-wheels/promises";
|
import { fireAndForget } from "octagonal-wheels/promises";
|
||||||
import { REMOTE_MINIO, REMOTE_P2P, type RemoteDBSettings } from "../../lib/src/common/types";
|
import { REMOTE_MINIO, REMOTE_P2P, type RemoteDBSettings } from "@lib/common/types";
|
||||||
import { LiveSyncCouchDBReplicator } from "../../lib/src/replication/couchdb/LiveSyncReplicator";
|
import { LiveSyncCouchDBReplicator } from "@lib/replication/couchdb/LiveSyncReplicator";
|
||||||
import type { LiveSyncAbstractReplicator } from "../../lib/src/replication/LiveSyncAbstractReplicator";
|
import type { LiveSyncAbstractReplicator } from "@lib/replication/LiveSyncAbstractReplicator";
|
||||||
import { AbstractModule } from "../AbstractModule";
|
import { AbstractModule } from "@/modules/AbstractModule";
|
||||||
import type { LiveSyncCore } from "../../main";
|
import type { LiveSyncCore } from "@/main";
|
||||||
|
|
||||||
export class ModuleReplicatorCouchDB extends AbstractModule {
|
export class ModuleReplicatorCouchDB extends AbstractModule {
|
||||||
_anyNewReplicator(settingOverride: Partial<RemoteDBSettings> = {}): Promise<LiveSyncAbstractReplicator | false> {
|
_anyNewReplicator(settingOverride: Partial<RemoteDBSettings> = {}): Promise<LiveSyncAbstractReplicator | false> {
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
import { REMOTE_MINIO, type RemoteDBSettings } from "../../lib/src/common/types";
|
import { REMOTE_MINIO, type RemoteDBSettings } from "@lib/common/types";
|
||||||
import { LiveSyncJournalReplicator } from "../../lib/src/replication/journal/LiveSyncJournalReplicator";
|
import { LiveSyncJournalReplicator } from "@lib/replication/journal/LiveSyncJournalReplicator";
|
||||||
import type { LiveSyncAbstractReplicator } from "../../lib/src/replication/LiveSyncAbstractReplicator";
|
import type { LiveSyncAbstractReplicator } from "@lib/replication/LiveSyncAbstractReplicator";
|
||||||
import type { LiveSyncCore } from "../../main";
|
import type { LiveSyncCore } from "@/main";
|
||||||
import { AbstractModule } from "../AbstractModule";
|
import { AbstractModule } from "@/modules/AbstractModule";
|
||||||
|
|
||||||
export class ModuleReplicatorMinIO extends AbstractModule {
|
export class ModuleReplicatorMinIO extends AbstractModule {
|
||||||
_anyNewReplicator(settingOverride: Partial<RemoteDBSettings> = {}): Promise<LiveSyncAbstractReplicator | false> {
|
_anyNewReplicator(settingOverride: Partial<RemoteDBSettings> = {}): Promise<LiveSyncAbstractReplicator | false> {
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ import {
|
|||||||
type MetaEntry,
|
type MetaEntry,
|
||||||
} from "@lib/common/types";
|
} from "@lib/common/types";
|
||||||
import type { ModuleReplicator } from "./ModuleReplicator";
|
import type { ModuleReplicator } from "./ModuleReplicator";
|
||||||
import { isChunk } from "@/lib/src/common/typeUtils";
|
import { isChunk } from "@lib/common/typeUtils";
|
||||||
import {
|
import {
|
||||||
LOG_LEVEL_DEBUG,
|
LOG_LEVEL_DEBUG,
|
||||||
LOG_LEVEL_INFO,
|
LOG_LEVEL_INFO,
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
import { AbstractModule } from "../AbstractModule.ts";
|
import { AbstractModule } from "@/modules/AbstractModule.ts";
|
||||||
import { LOG_LEVEL_NOTICE, type FilePathWithPrefix } from "../../lib/src/common/types";
|
import { LOG_LEVEL_NOTICE, type FilePathWithPrefix } from "@lib/common/types";
|
||||||
import { QueueProcessor } from "octagonal-wheels/concurrency/processor";
|
import { QueueProcessor } from "octagonal-wheels/concurrency/processor";
|
||||||
import { sendValue } from "octagonal-wheels/messagepassing/signal";
|
import { sendValue } from "octagonal-wheels/messagepassing/signal";
|
||||||
import type { InjectableServiceHub } from "../../lib/src/services/InjectableServices.ts";
|
import type { InjectableServiceHub } from "@lib/services/InjectableServices.ts";
|
||||||
import type { LiveSyncCore } from "../../main.ts";
|
import type { LiveSyncCore } from "@/main.ts";
|
||||||
|
|
||||||
export class ModuleConflictChecker extends AbstractModule {
|
export class ModuleConflictChecker extends AbstractModule {
|
||||||
async _queueConflictCheckIfOpen(file: FilePathWithPrefix): Promise<void> {
|
async _queueConflictCheckIfOpen(file: FilePathWithPrefix): Promise<void> {
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { serialized } from "octagonal-wheels/concurrency/lock";
|
import { serialized } from "octagonal-wheels/concurrency/lock";
|
||||||
import { AbstractModule } from "../AbstractModule.ts";
|
import { AbstractModule } from "@/modules/AbstractModule.ts";
|
||||||
import {
|
import {
|
||||||
AUTO_MERGED,
|
AUTO_MERGED,
|
||||||
CANCELLED,
|
CANCELLED,
|
||||||
@@ -10,15 +10,15 @@ import {
|
|||||||
NOT_CONFLICTED,
|
NOT_CONFLICTED,
|
||||||
type diff_check_result,
|
type diff_check_result,
|
||||||
type FilePathWithPrefix,
|
type FilePathWithPrefix,
|
||||||
} from "../../lib/src/common/types";
|
} from "@lib/common/types";
|
||||||
import { isCustomisationSyncMetadata, isPluginMetadata } from "@lib/common/typeUtils.ts";
|
import { isCustomisationSyncMetadata, isPluginMetadata } from "@lib/common/typeUtils.ts";
|
||||||
import { TARGET_IS_NEW } from "@lib/common/models/shared.const.symbols.ts";
|
import { TARGET_IS_NEW } from "@lib/common/models/shared.const.symbols.ts";
|
||||||
import { compareMTime, displayRev } from "@lib/common/utils.ts";
|
import { compareMTime, displayRev } from "@lib/common/utils.ts";
|
||||||
import diff_match_patch from "diff-match-patch";
|
import diff_match_patch from "diff-match-patch";
|
||||||
import { stripAllPrefixes, isPlainText } from "../../lib/src/string_and_binary/path";
|
import { stripAllPrefixes, isPlainText } from "@lib/string_and_binary/path";
|
||||||
import { eventHub } from "../../common/events.ts";
|
import { eventHub } from "@/common/events.ts";
|
||||||
import type { InjectableServiceHub } from "../../lib/src/services/InjectableServices.ts";
|
import type { InjectableServiceHub } from "@lib/services/InjectableServices.ts";
|
||||||
import type { LiveSyncCore } from "../../main.ts";
|
import type { LiveSyncCore } from "@/main.ts";
|
||||||
|
|
||||||
declare global {
|
declare global {
|
||||||
interface LSEvents {
|
interface LSEvents {
|
||||||
@@ -182,9 +182,9 @@ export class ModuleConflictResolver extends AbstractModule {
|
|||||||
revs.map(async (rev) => {
|
revs.map(async (rev) => {
|
||||||
const leaf = await this.core.databaseFileAccess.fetchEntryMeta(filename, rev);
|
const leaf = await this.core.databaseFileAccess.fetchEntryMeta(filename, rev);
|
||||||
if (leaf == false) {
|
if (leaf == false) {
|
||||||
return [0, rev] as [number, string];
|
return [0, rev];
|
||||||
}
|
}
|
||||||
return [leaf.mtime, rev] as [number, string];
|
return [leaf.mtime, rev];
|
||||||
})
|
})
|
||||||
)),
|
)),
|
||||||
] as [number, string][]
|
] as [number, string][]
|
||||||
|
|||||||
@@ -10,12 +10,12 @@ import {
|
|||||||
type RemoteDBSettings,
|
type RemoteDBSettings,
|
||||||
IncompatibleChangesInSpecificPattern,
|
IncompatibleChangesInSpecificPattern,
|
||||||
CompatibleButLossyChanges,
|
CompatibleButLossyChanges,
|
||||||
} from "../../lib/src/common/types.ts";
|
} from "@lib/common/types.ts";
|
||||||
import { escapeMarkdownValue } from "../../lib/src/common/utils.ts";
|
import { escapeMarkdownValue } from "@lib/common/utils.ts";
|
||||||
import { AbstractModule } from "../AbstractModule.ts";
|
import { AbstractModule } from "@/modules/AbstractModule.ts";
|
||||||
import { $msg } from "../../lib/src/common/i18n.ts";
|
import { $msg } from "@lib/common/i18n.ts";
|
||||||
import type { InjectableServiceHub } from "../../lib/src/services/InjectableServices.ts";
|
import type { InjectableServiceHub } from "@lib/services/InjectableServices.ts";
|
||||||
import type { LiveSyncCore } from "../../main.ts";
|
import type { LiveSyncCore } from "@/main.ts";
|
||||||
import { REMOTE_P2P } from "@lib/common/models/setting.const.ts";
|
import { REMOTE_P2P } from "@lib/common/models/setting.const.ts";
|
||||||
|
|
||||||
function valueToString(value: any) {
|
function valueToString(value: any) {
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { ButtonComponent } from "@/deps.ts";
|
import { ButtonComponent } from "@/deps.ts";
|
||||||
import { App, FuzzySuggestModal, MarkdownRenderer, Modal, Plugin, Setting } from "../../../deps.ts";
|
import { App, FuzzySuggestModal, MarkdownRenderer, Modal, Plugin, Setting } from "@/deps.ts";
|
||||||
import { EVENT_PLUGIN_UNLOADED, eventHub } from "../../../common/events.ts";
|
import { EVENT_PLUGIN_UNLOADED, eventHub } from "@/common/events.ts";
|
||||||
import { compatGlobal, type CompatIntervalHandle } from "@lib/common/coreEnvFunctions.ts";
|
import { compatGlobal, type CompatIntervalHandle } from "@lib/common/coreEnvFunctions.ts";
|
||||||
|
|
||||||
class AutoClosableModal extends Modal {
|
class AutoClosableModal extends Modal {
|
||||||
@@ -192,8 +192,10 @@ export class MessageBox<T extends readonly string[]> extends AutoClosableModal {
|
|||||||
const { contentEl } = this;
|
const { contentEl } = this;
|
||||||
this.titleEl.setText(this.title);
|
this.titleEl.setText(this.title);
|
||||||
const div = contentEl.createDiv();
|
const div = contentEl.createDiv();
|
||||||
div.style.userSelect = "text";
|
div.setCssStyles({
|
||||||
div.style["webkitUserSelect"] = "text";
|
userSelect: "text",
|
||||||
|
webkitUserSelect: "text",
|
||||||
|
});
|
||||||
void MarkdownRenderer.render(this.plugin.app, this.contentMd, div, "/", this.plugin);
|
void MarkdownRenderer.render(this.plugin.app, this.contentMd, div, "/", this.plugin);
|
||||||
const buttonSetting = new Setting(contentEl);
|
const buttonSetting = new Setting(contentEl);
|
||||||
const labelWrapper = contentEl.createDiv();
|
const labelWrapper = contentEl.createDiv();
|
||||||
@@ -202,21 +204,23 @@ export class MessageBox<T extends readonly string[]> extends AutoClosableModal {
|
|||||||
labelEl.addClass("sls-dialogue-note-countdown");
|
labelEl.addClass("sls-dialogue-note-countdown");
|
||||||
if (!this.timeout || !this.timer) {
|
if (!this.timeout || !this.timer) {
|
||||||
labelWrapper.empty();
|
labelWrapper.empty();
|
||||||
labelWrapper.style.display = "none";
|
labelWrapper.setCssStyles({ display: "none" });
|
||||||
}
|
}
|
||||||
|
|
||||||
buttonSetting.infoEl.style.display = "none";
|
buttonSetting.infoEl.setCssStyles({ display: "none" });
|
||||||
buttonSetting.controlEl.style.flexWrap = "wrap";
|
buttonSetting.controlEl.setCssStyles({ flexWrap: "wrap" });
|
||||||
if (this.wideButton) {
|
if (this.wideButton) {
|
||||||
buttonSetting.controlEl.style.flexDirection = "column";
|
buttonSetting.controlEl.setCssStyles({
|
||||||
buttonSetting.controlEl.style.alignItems = "center";
|
flexDirection: "column",
|
||||||
buttonSetting.controlEl.style.justifyContent = "center";
|
alignItems: "center",
|
||||||
buttonSetting.controlEl.style.flexGrow = "1";
|
justifyContent: "center",
|
||||||
|
flexGrow: "1",
|
||||||
|
});
|
||||||
}
|
}
|
||||||
contentEl.addEventListener("click", () => {
|
contentEl.addEventListener("click", () => {
|
||||||
if (this.timer) {
|
if (this.timer) {
|
||||||
labelWrapper.empty();
|
labelWrapper.empty();
|
||||||
labelWrapper.style.display = "none";
|
labelWrapper.setCssStyles({ display: "none" });
|
||||||
compatGlobal.clearInterval(this.timer);
|
compatGlobal.clearInterval(this.timer);
|
||||||
this.timer = undefined;
|
this.timer = undefined;
|
||||||
this.defaultButtonComponent?.setButtonText(`${this.defaultAction}`);
|
this.defaultButtonComponent?.setButtonText(`${this.defaultAction}`);
|
||||||
@@ -238,8 +242,10 @@ export class MessageBox<T extends readonly string[]> extends AutoClosableModal {
|
|||||||
btn.setCta();
|
btn.setCta();
|
||||||
}
|
}
|
||||||
if (this.wideButton) {
|
if (this.wideButton) {
|
||||||
btn.buttonEl.style.flexGrow = "1";
|
btn.buttonEl.setCssStyles({
|
||||||
btn.buttonEl.style.width = "100%";
|
flexGrow: "1",
|
||||||
|
width: "100%",
|
||||||
|
});
|
||||||
}
|
}
|
||||||
return btn;
|
return btn;
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,10 +1,10 @@
|
|||||||
// Obsidian to LiveSync Utils
|
// Obsidian to LiveSync Utils
|
||||||
|
|
||||||
import { TFile, type TAbstractFile, type TFolder } from "../../../deps.ts";
|
import { TFile, type TAbstractFile, type TFolder } from "@/deps.ts";
|
||||||
import { ICHeader } from "../../../common/types.ts";
|
import { ICHeader } from "@/common/types.ts";
|
||||||
import { addPrefix, isPlainText } from "../../../lib/src/string_and_binary/path.ts";
|
import { addPrefix, isPlainText } from "@lib/string_and_binary/path.ts";
|
||||||
import { LOG_LEVEL_VERBOSE, Logger } from "octagonal-wheels/common/logger";
|
import { LOG_LEVEL_VERBOSE, Logger } from "octagonal-wheels/common/logger";
|
||||||
import { createBlob } from "../../../lib/src/common/utils.ts";
|
import { createBlob } from "@lib/common/utils.ts";
|
||||||
import type {
|
import type {
|
||||||
FilePath,
|
FilePath,
|
||||||
FilePathWithPrefix,
|
FilePathWithPrefix,
|
||||||
@@ -12,8 +12,8 @@ import type {
|
|||||||
UXFileInfoStub,
|
UXFileInfoStub,
|
||||||
UXFolderInfo,
|
UXFolderInfo,
|
||||||
UXInternalFileInfoStub,
|
UXInternalFileInfoStub,
|
||||||
} from "../../../lib/src/common/types.ts";
|
} from "@lib/common/types.ts";
|
||||||
import type { LiveSyncCore } from "../../../main.ts";
|
import type { LiveSyncCore } from "@/main.ts";
|
||||||
import type { FileAccessObsidian } from "@/serviceModules/FileAccessObsidian.ts";
|
import type { FileAccessObsidian } from "@/serviceModules/FileAccessObsidian.ts";
|
||||||
|
|
||||||
export async function TFileToUXFileInfo(
|
export async function TFileToUXFileInfo(
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import type { LiveSyncCore } from "@/main";
|
import type { LiveSyncCore } from "@/main";
|
||||||
import { LOG_LEVEL_NOTICE } from "octagonal-wheels/common/logger";
|
import { LOG_LEVEL_NOTICE } from "octagonal-wheels/common/logger";
|
||||||
import { fireAndForget } from "octagonal-wheels/promises";
|
import { fireAndForget } from "octagonal-wheels/promises";
|
||||||
import { AbstractModule } from "../AbstractModule";
|
import { AbstractModule } from "@/modules/AbstractModule";
|
||||||
// Separated Module for basic menu commands, which are not related to obsidian specific features. It is expected to be used in other platforms with minimal changes.
|
// Separated Module for basic menu commands, which are not related to obsidian specific features. It is expected to be used in other platforms with minimal changes.
|
||||||
// However, it is odd that it has here at all; it really ought to be in each respective feature. It will likely be moved eventually. Until now, addCommand pointed to Obsidian's version.
|
// However, it is odd that it has here at all; it really ought to be in each respective feature. It will likely be moved eventually. Until now, addCommand pointed to Obsidian's version.
|
||||||
export class ModuleBasicMenu extends AbstractModule {
|
export class ModuleBasicMenu extends AbstractModule {
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { LOG_LEVEL_NOTICE, LOG_LEVEL_VERBOSE, Logger } from "../../lib/src/common/logger.ts";
|
import { LOG_LEVEL_NOTICE, LOG_LEVEL_VERBOSE, Logger } from "@lib/common/logger.ts";
|
||||||
import {
|
import {
|
||||||
EVENT_REQUEST_OPEN_P2P,
|
EVENT_REQUEST_OPEN_P2P,
|
||||||
EVENT_REQUEST_OPEN_SETTING_WIZARD,
|
EVENT_REQUEST_OPEN_SETTING_WIZARD,
|
||||||
@@ -6,16 +6,16 @@ import {
|
|||||||
EVENT_REQUEST_RUN_DOCTOR,
|
EVENT_REQUEST_RUN_DOCTOR,
|
||||||
EVENT_REQUEST_RUN_FIX_INCOMPLETE,
|
EVENT_REQUEST_RUN_FIX_INCOMPLETE,
|
||||||
eventHub,
|
eventHub,
|
||||||
} from "../../common/events.ts";
|
} from "@/common/events.ts";
|
||||||
import { AbstractModule } from "../AbstractModule.ts";
|
import { AbstractModule } from "@/modules/AbstractModule.ts";
|
||||||
import { $msg } from "@lib/common/i18n.ts";
|
import { $msg } from "@lib/common/i18n.ts";
|
||||||
import { performDoctorConsultation, RebuildOptions } from "../../lib/src/common/configForDoc.ts";
|
import { performDoctorConsultation, RebuildOptions } from "@lib/common/configForDoc.ts";
|
||||||
import { isValidPath } from "../../common/utils.ts";
|
import { isValidPath } from "@/common/utils.ts";
|
||||||
import { isMetaEntry } from "../../lib/src/common/types.ts";
|
import { isMetaEntry } from "@lib/common/types.ts";
|
||||||
import { isDeletedEntry, isDocContentSame, isLoadedEntry, readAsBlob } from "../../lib/src/common/utils.ts";
|
import { isDeletedEntry, isDocContentSame, isLoadedEntry, readAsBlob } from "@lib/common/utils.ts";
|
||||||
import { countCompromisedChunks } from "../../lib/src/pouchdb/negotiation.ts";
|
import { countCompromisedChunks } from "@lib/pouchdb/negotiation.ts";
|
||||||
import type { LiveSyncCore } from "../../main.ts";
|
import type { LiveSyncCore } from "@/main.ts";
|
||||||
import { SetupManager } from "../features/SetupManager.ts";
|
import { SetupManager } from "@/modules/features/SetupManager.ts";
|
||||||
|
|
||||||
type ErrorInfo = {
|
type ErrorInfo = {
|
||||||
path: string;
|
path: string;
|
||||||
|
|||||||
@@ -5,14 +5,26 @@
|
|||||||
|
|
||||||
import { FetchHttpHandler, type FetchHttpHandlerOptions } from "@smithy/fetch-http-handler";
|
import { FetchHttpHandler, type FetchHttpHandlerOptions } from "@smithy/fetch-http-handler";
|
||||||
import { HttpRequest, HttpResponse, type HttpHandlerOptions } from "@smithy/protocol-http";
|
import { HttpRequest, HttpResponse, type HttpHandlerOptions } from "@smithy/protocol-http";
|
||||||
//@ts-ignore
|
|
||||||
import { requestTimeout } from "@smithy/fetch-http-handler/dist-es/request-timeout";
|
|
||||||
import { buildQueryString } from "@smithy/querystring-builder";
|
import { buildQueryString } from "@smithy/querystring-builder";
|
||||||
import { requestUrl, type RequestUrlParam } from "../../../deps.ts";
|
import { requestUrl, type RequestUrlParam } from "@/deps.ts";
|
||||||
|
import { compatGlobal } from "@lib/common/coreEnvFunctions.ts";
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// special handler using Obsidian requestUrl
|
// special handler using Obsidian requestUrl
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
function requestTimeout(timeoutInMs: number = 0): Promise<never> {
|
||||||
|
return new Promise((_, reject) => {
|
||||||
|
if (timeoutInMs) {
|
||||||
|
compatGlobal.setTimeout(() => {
|
||||||
|
const timeoutError = new Error(`Request did not complete within ${timeoutInMs} ms`);
|
||||||
|
timeoutError.name = "TimeoutError";
|
||||||
|
reject(timeoutError);
|
||||||
|
}, timeoutInMs);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This is close to origin implementation of FetchHttpHandler
|
* This is close to origin implementation of FetchHttpHandler
|
||||||
* https://github.com/aws/aws-sdk-js-v3/blob/main/packages/fetch-http-handler/src/fetch-http-handler.ts
|
* https://github.com/aws/aws-sdk-js-v3/blob/main/packages/fetch-http-handler/src/fetch-http-handler.ts
|
||||||
|
|||||||
@@ -1,18 +1,18 @@
|
|||||||
import { AbstractObsidianModule } from "../AbstractObsidianModule.ts";
|
import { AbstractObsidianModule } from "@/modules/AbstractObsidianModule.ts";
|
||||||
import { EVENT_FILE_RENAMED, EVENT_LEAF_ACTIVE_CHANGED, eventHub } from "../../common/events.js";
|
import { EVENT_FILE_RENAMED, EVENT_LEAF_ACTIVE_CHANGED, eventHub } from "@/common/events.js";
|
||||||
import { LOG_LEVEL_NOTICE, LOG_LEVEL_VERBOSE } from "octagonal-wheels/common/logger";
|
import { LOG_LEVEL_NOTICE, LOG_LEVEL_VERBOSE } from "octagonal-wheels/common/logger";
|
||||||
import { scheduleTask } from "octagonal-wheels/concurrency/task";
|
import { scheduleTask } from "octagonal-wheels/concurrency/task";
|
||||||
import type { TFile } from "../../deps.ts";
|
import type { TFile } from "@/deps.ts";
|
||||||
import { fireAndForget } from "octagonal-wheels/promises";
|
import { fireAndForget } from "octagonal-wheels/promises";
|
||||||
import { type FilePathWithPrefix } from "../../lib/src/common/types.ts";
|
import { type FilePathWithPrefix } from "@lib/common/types.ts";
|
||||||
import { reactive, reactiveSource, type ReactiveSource } from "octagonal-wheels/dataobject/reactive";
|
import { reactive, reactiveSource, type ReactiveSource } from "octagonal-wheels/dataobject/reactive";
|
||||||
import {
|
import {
|
||||||
collectingChunks,
|
collectingChunks,
|
||||||
pluginScanningCount,
|
pluginScanningCount,
|
||||||
hiddenFilesEventCount,
|
hiddenFilesEventCount,
|
||||||
hiddenFilesProcessingCount,
|
hiddenFilesProcessingCount,
|
||||||
} from "../../lib/src/mock_and_interop/stores.ts";
|
} from "@lib/mock_and_interop/stores.ts";
|
||||||
import type { LiveSyncCore } from "../../main.ts";
|
import type { LiveSyncCore } from "@/main.ts";
|
||||||
import { compatGlobal } from "@lib/common/coreEnvFunctions.ts";
|
import { compatGlobal } from "@lib/common/coreEnvFunctions.ts";
|
||||||
|
|
||||||
export class ModuleObsidianEvents extends AbstractObsidianModule {
|
export class ModuleObsidianEvents extends AbstractObsidianModule {
|
||||||
@@ -63,12 +63,12 @@ export class ModuleObsidianEvents extends AbstractObsidianModule {
|
|||||||
// eslint-disable-next-line @typescript-eslint/no-this-alias
|
// eslint-disable-next-line @typescript-eslint/no-this-alias
|
||||||
const _this = this;
|
const _this = this;
|
||||||
//@ts-ignore
|
//@ts-ignore
|
||||||
if (!window.CodeMirrorAdapter) {
|
if (!compatGlobal.CodeMirrorAdapter) {
|
||||||
this._log("CodeMirrorAdapter is not available");
|
this._log("CodeMirrorAdapter is not available");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
//@ts-ignore
|
//@ts-ignore
|
||||||
window.CodeMirrorAdapter.commands.save = () => {
|
compatGlobal.CodeMirrorAdapter.commands.save = () => {
|
||||||
//@ts-ignore
|
//@ts-ignore
|
||||||
void _this.app.commands.executeCommandById("editor:save-file");
|
void _this.app.commands.executeCommandById("editor:save-file");
|
||||||
// _this.app.performCommand('editor:save-file');
|
// _this.app.performCommand('editor:save-file');
|
||||||
@@ -86,14 +86,14 @@ export class ModuleObsidianEvents extends AbstractObsidianModule {
|
|||||||
// Already bound
|
// Already bound
|
||||||
// eslint-disable-next-line @typescript-eslint/unbound-method
|
// eslint-disable-next-line @typescript-eslint/unbound-method
|
||||||
this.plugin.registerDomEvent(activeDocument, "visibilitychange", this.watchWindowVisibility);
|
this.plugin.registerDomEvent(activeDocument, "visibilitychange", this.watchWindowVisibility);
|
||||||
this.plugin.registerDomEvent(window, "focus", () => this.setHasFocus(true));
|
this.plugin.registerDomEvent(compatGlobal, "focus", () => this.setHasFocus(true));
|
||||||
this.plugin.registerDomEvent(window, "blur", () => this.setHasFocus(false));
|
this.plugin.registerDomEvent(compatGlobal, "blur", () => this.setHasFocus(false));
|
||||||
// Already bound
|
// Already bound
|
||||||
// eslint-disable-next-line @typescript-eslint/unbound-method
|
// eslint-disable-next-line @typescript-eslint/unbound-method
|
||||||
this.plugin.registerDomEvent(window, "online", this.watchOnline);
|
this.plugin.registerDomEvent(compatGlobal, "online", this.watchOnline);
|
||||||
// Already bound
|
// Already bound
|
||||||
// eslint-disable-next-line @typescript-eslint/unbound-method
|
// eslint-disable-next-line @typescript-eslint/unbound-method
|
||||||
this.plugin.registerDomEvent(window, "offline", this.watchOnline);
|
this.plugin.registerDomEvent(compatGlobal, "offline", this.watchOnline);
|
||||||
}
|
}
|
||||||
|
|
||||||
hasFocus = true;
|
hasFocus = true;
|
||||||
@@ -114,7 +114,7 @@ export class ModuleObsidianEvents extends AbstractObsidianModule {
|
|||||||
async watchOnlineAsync() {
|
async watchOnlineAsync() {
|
||||||
// If some files were failed to retrieve, scan files again.
|
// If some files were failed to retrieve, scan files again.
|
||||||
// TODO:FIXME AT V0.17.31, this logic has been disabled.
|
// TODO:FIXME AT V0.17.31, this logic has been disabled.
|
||||||
if (navigator.onLine && this.localDatabase.needScanning) {
|
if (compatGlobal.navigator.onLine && this.localDatabase.needScanning) {
|
||||||
this.localDatabase.needScanning = false;
|
this.localDatabase.needScanning = false;
|
||||||
await this.services.vault.scanVault();
|
await this.services.vault.scanVault();
|
||||||
}
|
}
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user