pro and smart conflict
This commit is contained in:
parent
0802767726
commit
06dad54d4c
@ -1,3 +1,5 @@
|
|||||||
DROPBOX_APP_KEY=
|
DROPBOX_APP_KEY=
|
||||||
ONEDRIVE_CLIENT_ID=
|
ONEDRIVE_CLIENT_ID=
|
||||||
ONEDRIVE_AUTHORITY=https://
|
ONEDRIVE_AUTHORITY=https://
|
||||||
|
REMOTELYSAVE_WEBSITE=http://127.0.0.1:46683
|
||||||
|
REMOTELYSAVE_CLIENT_ID=cli-xxx
|
||||||
|
|||||||
203
LICENSE
203
LICENSE
@ -1,202 +1,3 @@
|
|||||||
Apache License
|
The codes or files or subfolders inside the folder `src`, `tests`, `docs`, `assets`, are released under the "Open Source" license: "Apache License, version 2.0", described at: https://www.apache.org/licenses/LICENSE-2.0 .
|
||||||
Version 2.0, January 2004
|
|
||||||
http://www.apache.org/licenses/
|
|
||||||
|
|
||||||
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
|
||||||
|
|
||||||
1. Definitions.
|
|
||||||
|
|
||||||
"License" shall mean the terms and conditions for use, reproduction,
|
|
||||||
and distribution as defined by Sections 1 through 9 of this document.
|
|
||||||
|
|
||||||
"Licensor" shall mean the copyright owner or entity authorized by
|
|
||||||
the copyright owner that is granting the License.
|
|
||||||
|
|
||||||
"Legal Entity" shall mean the union of the acting entity and all
|
|
||||||
other entities that control, are controlled by, or are under common
|
|
||||||
control with that entity. For the purposes of this definition,
|
|
||||||
"control" means (i) the power, direct or indirect, to cause the
|
|
||||||
direction or management of such entity, whether by contract or
|
|
||||||
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
|
||||||
outstanding shares, or (iii) beneficial ownership of such entity.
|
|
||||||
|
|
||||||
"You" (or "Your") shall mean an individual or Legal Entity
|
|
||||||
exercising permissions granted by this License.
|
|
||||||
|
|
||||||
"Source" form shall mean the preferred form for making modifications,
|
|
||||||
including but not limited to software source code, documentation
|
|
||||||
source, and configuration files.
|
|
||||||
|
|
||||||
"Object" form shall mean any form resulting from mechanical
|
|
||||||
transformation or translation of a Source form, including but
|
|
||||||
not limited to compiled object code, generated documentation,
|
|
||||||
and conversions to other media types.
|
|
||||||
|
|
||||||
"Work" shall mean the work of authorship, whether in Source or
|
|
||||||
Object form, made available under the License, as indicated by a
|
|
||||||
copyright notice that is included in or attached to the work
|
|
||||||
(an example is provided in the Appendix below).
|
|
||||||
|
|
||||||
"Derivative Works" shall mean any work, whether in Source or Object
|
|
||||||
form, that is based on (or derived from) the Work and for which the
|
|
||||||
editorial revisions, annotations, elaborations, or other modifications
|
|
||||||
represent, as a whole, an original work of authorship. For the purposes
|
|
||||||
of this License, Derivative Works shall not include works that remain
|
|
||||||
separable from, or merely link (or bind by name) to the interfaces of,
|
|
||||||
the Work and Derivative Works thereof.
|
|
||||||
|
|
||||||
"Contribution" shall mean any work of authorship, including
|
|
||||||
the original version of the Work and any modifications or additions
|
|
||||||
to that Work or Derivative Works thereof, that is intentionally
|
|
||||||
submitted to Licensor for inclusion in the Work by the copyright owner
|
|
||||||
or by an individual or Legal Entity authorized to submit on behalf of
|
|
||||||
the copyright owner. For the purposes of this definition, "submitted"
|
|
||||||
means any form of electronic, verbal, or written communication sent
|
|
||||||
to the Licensor or its representatives, including but not limited to
|
|
||||||
communication on electronic mailing lists, source code control systems,
|
|
||||||
and issue tracking systems that are managed by, or on behalf of, the
|
|
||||||
Licensor for the purpose of discussing and improving the Work, but
|
|
||||||
excluding communication that is conspicuously marked or otherwise
|
|
||||||
designated in writing by the copyright owner as "Not a Contribution."
|
|
||||||
|
|
||||||
"Contributor" shall mean Licensor and any individual or Legal Entity
|
|
||||||
on behalf of whom a Contribution has been received by Licensor and
|
|
||||||
subsequently incorporated within the Work.
|
|
||||||
|
|
||||||
2. Grant of Copyright License. Subject to the terms and conditions of
|
|
||||||
this License, each Contributor hereby grants to You a perpetual,
|
|
||||||
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
|
||||||
copyright license to reproduce, prepare Derivative Works of,
|
|
||||||
publicly display, publicly perform, sublicense, and distribute the
|
|
||||||
Work and such Derivative Works in Source or Object form.
|
|
||||||
|
|
||||||
3. Grant of Patent License. Subject to the terms and conditions of
|
|
||||||
this License, each Contributor hereby grants to You a perpetual,
|
|
||||||
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
|
||||||
(except as stated in this section) patent license to make, have made,
|
|
||||||
use, offer to sell, sell, import, and otherwise transfer the Work,
|
|
||||||
where such license applies only to those patent claims licensable
|
|
||||||
by such Contributor that are necessarily infringed by their
|
|
||||||
Contribution(s) alone or by combination of their Contribution(s)
|
|
||||||
with the Work to which such Contribution(s) was submitted. If You
|
|
||||||
institute patent litigation against any entity (including a
|
|
||||||
cross-claim or counterclaim in a lawsuit) alleging that the Work
|
|
||||||
or a Contribution incorporated within the Work constitutes direct
|
|
||||||
or contributory patent infringement, then any patent licenses
|
|
||||||
granted to You under this License for that Work shall terminate
|
|
||||||
as of the date such litigation is filed.
|
|
||||||
|
|
||||||
4. Redistribution. You may reproduce and distribute copies of the
|
|
||||||
Work or Derivative Works thereof in any medium, with or without
|
|
||||||
modifications, and in Source or Object form, provided that You
|
|
||||||
meet the following conditions:
|
|
||||||
|
|
||||||
(a) You must give any other recipients of the Work or
|
|
||||||
Derivative Works a copy of this License; and
|
|
||||||
|
|
||||||
(b) You must cause any modified files to carry prominent notices
|
|
||||||
stating that You changed the files; and
|
|
||||||
|
|
||||||
(c) You must retain, in the Source form of any Derivative Works
|
|
||||||
that You distribute, all copyright, patent, trademark, and
|
|
||||||
attribution notices from the Source form of the Work,
|
|
||||||
excluding those notices that do not pertain to any part of
|
|
||||||
the Derivative Works; and
|
|
||||||
|
|
||||||
(d) If the Work includes a "NOTICE" text file as part of its
|
|
||||||
distribution, then any Derivative Works that You distribute must
|
|
||||||
include a readable copy of the attribution notices contained
|
|
||||||
within such NOTICE file, excluding those notices that do not
|
|
||||||
pertain to any part of the Derivative Works, in at least one
|
|
||||||
of the following places: within a NOTICE text file distributed
|
|
||||||
as part of the Derivative Works; within the Source form or
|
|
||||||
documentation, if provided along with the Derivative Works; or,
|
|
||||||
within a display generated by the Derivative Works, if and
|
|
||||||
wherever such third-party notices normally appear. The contents
|
|
||||||
of the NOTICE file are for informational purposes only and
|
|
||||||
do not modify the License. You may add Your own attribution
|
|
||||||
notices within Derivative Works that You distribute, alongside
|
|
||||||
or as an addendum to the NOTICE text from the Work, provided
|
|
||||||
that such additional attribution notices cannot be construed
|
|
||||||
as modifying the License.
|
|
||||||
|
|
||||||
You may add Your own copyright statement to Your modifications and
|
|
||||||
may provide additional or different license terms and conditions
|
|
||||||
for use, reproduction, or distribution of Your modifications, or
|
|
||||||
for any such Derivative Works as a whole, provided Your use,
|
|
||||||
reproduction, and distribution of the Work otherwise complies with
|
|
||||||
the conditions stated in this License.
|
|
||||||
|
|
||||||
5. Submission of Contributions. Unless You explicitly state otherwise,
|
|
||||||
any Contribution intentionally submitted for inclusion in the Work
|
|
||||||
by You to the Licensor shall be under the terms and conditions of
|
|
||||||
this License, without any additional terms or conditions.
|
|
||||||
Notwithstanding the above, nothing herein shall supersede or modify
|
|
||||||
the terms of any separate license agreement you may have executed
|
|
||||||
with Licensor regarding such Contributions.
|
|
||||||
|
|
||||||
6. Trademarks. This License does not grant permission to use the trade
|
|
||||||
names, trademarks, service marks, or product names of the Licensor,
|
|
||||||
except as required for reasonable and customary use in describing the
|
|
||||||
origin of the Work and reproducing the content of the NOTICE file.
|
|
||||||
|
|
||||||
7. Disclaimer of Warranty. Unless required by applicable law or
|
|
||||||
agreed to in writing, Licensor provides the Work (and each
|
|
||||||
Contributor provides its Contributions) on an "AS IS" BASIS,
|
|
||||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
|
||||||
implied, including, without limitation, any warranties or conditions
|
|
||||||
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
|
|
||||||
PARTICULAR PURPOSE. You are solely responsible for determining the
|
|
||||||
appropriateness of using or redistributing the Work and assume any
|
|
||||||
risks associated with Your exercise of permissions under this License.
|
|
||||||
|
|
||||||
8. Limitation of Liability. In no event and under no legal theory,
|
|
||||||
whether in tort (including negligence), contract, or otherwise,
|
|
||||||
unless required by applicable law (such as deliberate and grossly
|
|
||||||
negligent acts) or agreed to in writing, shall any Contributor be
|
|
||||||
liable to You for damages, including any direct, indirect, special,
|
|
||||||
incidental, or consequential damages of any character arising as a
|
|
||||||
result of this License or out of the use or inability to use the
|
|
||||||
Work (including but not limited to damages for loss of goodwill,
|
|
||||||
work stoppage, computer failure or malfunction, or any and all
|
|
||||||
other commercial damages or losses), even if such Contributor
|
|
||||||
has been advised of the possibility of such damages.
|
|
||||||
|
|
||||||
9. Accepting Warranty or Additional Liability. While redistributing
|
|
||||||
the Work or Derivative Works thereof, You may choose to offer,
|
|
||||||
and charge a fee for, acceptance of support, warranty, indemnity,
|
|
||||||
or other liability obligations and/or rights consistent with this
|
|
||||||
License. However, in accepting such obligations, You may act only
|
|
||||||
on Your own behalf and on Your sole responsibility, not on behalf
|
|
||||||
of any other Contributor, and only if You agree to indemnify,
|
|
||||||
defend, and hold each Contributor harmless for any liability
|
|
||||||
incurred by, or claims asserted against, such Contributor by reason
|
|
||||||
of your accepting any such warranty or additional liability.
|
|
||||||
|
|
||||||
END OF TERMS AND CONDITIONS
|
|
||||||
|
|
||||||
APPENDIX: How to apply the Apache License to your work.
|
|
||||||
|
|
||||||
To apply the Apache License to your work, attach the following
|
|
||||||
boilerplate notice, with the fields enclosed by brackets "[]"
|
|
||||||
replaced with your own identifying information. (Don't include
|
|
||||||
the brackets!) The text should be enclosed in the appropriate
|
|
||||||
comment syntax for the file format. We also recommend that a
|
|
||||||
file or class name and description of purpose be included on the
|
|
||||||
same "printed page" as the copyright notice for easier
|
|
||||||
identification within third-party archives.
|
|
||||||
|
|
||||||
Copyright [yyyy] [name of copyright owner]
|
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
you may not use this file except in compliance with the License.
|
|
||||||
You may obtain a copy of the License at
|
|
||||||
|
|
||||||
http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
|
|
||||||
Unless required by applicable law or agreed to in writing, software
|
|
||||||
distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
See the License for the specific language governing permissions and
|
|
||||||
limitations under the License.
|
|
||||||
|
|
||||||
|
The codes or files or subfolders inside the folder `pro`, are released under the "Source Available" license: "PolyForm Strict License 1.0.0", described at: https://polyformproject.org/licenses/strict/1.0.0/ .
|
||||||
|
|||||||
10
README.md
10
README.md
@ -28,9 +28,9 @@ This is yet another unofficial sync plugin for Obsidian. If you like it or find
|
|||||||
- **Scheduled auto sync supported.** You can also manually trigger the sync using sidebar ribbon, or using the command from the command palette (or even bind the hot key combination to the command then press the hot key combination).
|
- **Scheduled auto sync supported.** You can also manually trigger the sync using sidebar ribbon, or using the command from the command palette (or even bind the hot key combination to the command then press the hot key combination).
|
||||||
- **[Minimal Intrusive](./docs/minimal_intrusive_design.md).**
|
- **[Minimal Intrusive](./docs/minimal_intrusive_design.md).**
|
||||||
- **Skip Large files** and **skip paths** by custom regex conditions!
|
- **Skip Large files** and **skip paths** by custom regex conditions!
|
||||||
- **Fully open source under [Apache-2.0 License](./LICENSE).**
|
- **[Sync Algorithm](./docs/sync_algorithm/v3/intro.md) is provided for discussion.**
|
||||||
- **[Sync Algorithm open](./docs/sync_algorithm/v3/intro.md) for discussion.**
|
- **[Basic Conflict Detection And Handling](./docs/sync_algorithm/v3/intro.md)** for free version. **[Advanced Conflict Handling](./pro/README.md)** for PRO version.
|
||||||
- **[Basic Conflict Detection And Handling](./docs/sync_algorithm/v3/intro.md)** now, more to come!
|
- Source Available. See [License](./LICENSE) for details.
|
||||||
|
|
||||||
## Limitations
|
## Limitations
|
||||||
|
|
||||||
@ -131,6 +131,10 @@ Additionally, the plugin author may occasionally visit Obsidian official forum a
|
|||||||
|
|
||||||
In the latest version, you can change the settings to allow syncing `_` files or folders, as well as `.obsidian` special config folder (but not any other `.` files or folders).
|
In the latest version, you can change the settings to allow syncing `_` files or folders, as well as `.obsidian` special config folder (but not any other `.` files or folders).
|
||||||
|
|
||||||
|
## PRO Features
|
||||||
|
|
||||||
|
See [PRO](./pro/README.md) for more details.
|
||||||
|
|
||||||
## How To Debug
|
## How To Debug
|
||||||
|
|
||||||
See [here](./docs/how_to_debug/README.md) for more details.
|
See [here](./docs/how_to_debug/README.md) for more details.
|
||||||
|
|||||||
@ -1,3 +1,4 @@
|
|||||||
|
import "dotenv/config";
|
||||||
import esbuild from "esbuild";
|
import esbuild from "esbuild";
|
||||||
import inlineWorkerPlugin from "esbuild-plugin-inline-worker";
|
import inlineWorkerPlugin from "esbuild-plugin-inline-worker";
|
||||||
import process from "process";
|
import process from "process";
|
||||||
@ -16,6 +17,8 @@ const prod = process.argv[2] === "production";
|
|||||||
const DEFAULT_DROPBOX_APP_KEY = process.env.DROPBOX_APP_KEY || "";
|
const DEFAULT_DROPBOX_APP_KEY = process.env.DROPBOX_APP_KEY || "";
|
||||||
const DEFAULT_ONEDRIVE_CLIENT_ID = process.env.ONEDRIVE_CLIENT_ID || "";
|
const DEFAULT_ONEDRIVE_CLIENT_ID = process.env.ONEDRIVE_CLIENT_ID || "";
|
||||||
const DEFAULT_ONEDRIVE_AUTHORITY = process.env.ONEDRIVE_AUTHORITY || "";
|
const DEFAULT_ONEDRIVE_AUTHORITY = process.env.ONEDRIVE_AUTHORITY || "";
|
||||||
|
const DEFAULT_REMOTELYSAVE_WEBSITE = process.env.REMOTELYSAVE_WEBSITE || "";
|
||||||
|
const DEFAULT_REMOTELYSAVE_CLIENT_ID = process.env.REMOTELYSAVE_CLIENT_ID || "";
|
||||||
|
|
||||||
esbuild
|
esbuild
|
||||||
.context({
|
.context({
|
||||||
@ -51,6 +54,8 @@ esbuild
|
|||||||
"process.env.DEFAULT_DROPBOX_APP_KEY": `"${DEFAULT_DROPBOX_APP_KEY}"`,
|
"process.env.DEFAULT_DROPBOX_APP_KEY": `"${DEFAULT_DROPBOX_APP_KEY}"`,
|
||||||
"process.env.DEFAULT_ONEDRIVE_CLIENT_ID": `"${DEFAULT_ONEDRIVE_CLIENT_ID}"`,
|
"process.env.DEFAULT_ONEDRIVE_CLIENT_ID": `"${DEFAULT_ONEDRIVE_CLIENT_ID}"`,
|
||||||
"process.env.DEFAULT_ONEDRIVE_AUTHORITY": `"${DEFAULT_ONEDRIVE_AUTHORITY}"`,
|
"process.env.DEFAULT_ONEDRIVE_AUTHORITY": `"${DEFAULT_ONEDRIVE_AUTHORITY}"`,
|
||||||
|
"process.env.DEFAULT_REMOTELYSAVE_WEBSITE": `"${DEFAULT_REMOTELYSAVE_WEBSITE}"`,
|
||||||
|
"process.env.DEFAULT_REMOTELYSAVE_CLIENT_ID": `"${DEFAULT_REMOTELYSAVE_CLIENT_ID}"`,
|
||||||
global: "window",
|
global: "window",
|
||||||
"process.env.NODE_DEBUG": `undefined`, // ugly fix
|
"process.env.NODE_DEBUG": `undefined`, // ugly fix
|
||||||
"process.env.DEBUG": `undefined`, // ugly fix
|
"process.env.DEBUG": `undefined`, // ugly fix
|
||||||
|
|||||||
@ -1,11 +1,11 @@
|
|||||||
{
|
{
|
||||||
"id": "remotely-save",
|
"id": "remotely-save",
|
||||||
"name": "Remotely Save",
|
"name": "Remotely Save",
|
||||||
"version": "0.4.25",
|
"version": "0.5.1",
|
||||||
"minAppVersion": "0.13.21",
|
"minAppVersion": "0.13.21",
|
||||||
"description": "Yet another unofficial plugin allowing users to synchronize notes between local device and the cloud service.",
|
"description": "Yet another unofficial plugin allowing users to synchronize notes between local device and the cloud service.",
|
||||||
"author": "fyears",
|
"author": "fyears",
|
||||||
"authorUrl": "https://github.com/fyears",
|
"authorUrl": "https://github.com/fyears",
|
||||||
"isDesktopOnly": false,
|
"isDesktopOnly": false,
|
||||||
"fundingUrl": "https://github.com/remotely-save/donation"
|
"fundingUrl": "https://remotelysave.com"
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,11 +1,11 @@
|
|||||||
{
|
{
|
||||||
"id": "remotely-save",
|
"id": "remotely-save",
|
||||||
"name": "Remotely Save",
|
"name": "Remotely Save",
|
||||||
"version": "0.4.25",
|
"version": "0.5.1",
|
||||||
"minAppVersion": "0.13.21",
|
"minAppVersion": "0.13.21",
|
||||||
"description": "Yet another unofficial plugin allowing users to synchronize notes between local device and the cloud service.",
|
"description": "Yet another unofficial plugin allowing users to synchronize notes between local device and the cloud service.",
|
||||||
"author": "fyears",
|
"author": "fyears",
|
||||||
"authorUrl": "https://github.com/fyears",
|
"authorUrl": "https://github.com/fyears",
|
||||||
"isDesktopOnly": false,
|
"isDesktopOnly": false,
|
||||||
"fundingUrl": "https://github.com/remotely-save/donation"
|
"fundingUrl": "https://remotelysave.com"
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "remotely-save",
|
"name": "remotely-save",
|
||||||
"version": "0.4.25",
|
"version": "0.5.1",
|
||||||
"description": "This is yet another sync plugin for Obsidian app.",
|
"description": "This is yet another sync plugin for Obsidian app.",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev2": "node esbuild.config.mjs --watch",
|
"dev2": "node esbuild.config.mjs --watch",
|
||||||
@ -9,7 +9,7 @@
|
|||||||
"dev": "webpack --mode development --watch",
|
"dev": "webpack --mode development --watch",
|
||||||
"format": "npx @biomejs/biome check --apply .",
|
"format": "npx @biomejs/biome check --apply .",
|
||||||
"clean": "npx rimraf main.js",
|
"clean": "npx rimraf main.js",
|
||||||
"test": "cross-env TS_NODE_COMPILER_OPTIONS={\\\"module\\\":\\\"commonjs\\\"} mocha -r ts-node/register 'tests/**/*.ts'"
|
"test": "cross-env TS_NODE_COMPILER_OPTIONS={\\\"module\\\":\\\"commonjs\\\"} mocha -r ts-node/register 'tests/**/*.ts' 'pro/tests/**/*.ts'"
|
||||||
},
|
},
|
||||||
"browser": {
|
"browser": {
|
||||||
"path": "path-browserify",
|
"path": "path-browserify",
|
||||||
@ -23,7 +23,7 @@
|
|||||||
"source": "main.ts",
|
"source": "main.ts",
|
||||||
"keywords": [],
|
"keywords": [],
|
||||||
"author": "",
|
"author": "",
|
||||||
"license": "Apache-2.0",
|
"license": "SEE LICENSE IN LICENSE",
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@biomejs/biome": "1.7.3",
|
"@biomejs/biome": "1.7.3",
|
||||||
"@microsoft/microsoft-graph-types": "^2.40.0",
|
"@microsoft/microsoft-graph-types": "^2.40.0",
|
||||||
@ -63,6 +63,7 @@
|
|||||||
"@fyears/rclone-crypt": "^0.0.7",
|
"@fyears/rclone-crypt": "^0.0.7",
|
||||||
"@fyears/tsqueue": "^1.0.1",
|
"@fyears/tsqueue": "^1.0.1",
|
||||||
"@microsoft/microsoft-graph-client": "^3.0.7",
|
"@microsoft/microsoft-graph-client": "^3.0.7",
|
||||||
|
"@sanity/diff-match-patch": "^3.1.1",
|
||||||
"@smithy/fetch-http-handler": "^2.5.0",
|
"@smithy/fetch-http-handler": "^2.5.0",
|
||||||
"@smithy/protocol-http": "^3.3.0",
|
"@smithy/protocol-http": "^3.3.0",
|
||||||
"@smithy/querystring-builder": "^2.2.0",
|
"@smithy/querystring-builder": "^2.2.0",
|
||||||
@ -83,6 +84,7 @@
|
|||||||
"mime-types": "^2.1.35",
|
"mime-types": "^2.1.35",
|
||||||
"mustache": "^4.2.0",
|
"mustache": "^4.2.0",
|
||||||
"nanoid": "^5.0.7",
|
"nanoid": "^5.0.7",
|
||||||
|
"node-diff3": "^3.1.2",
|
||||||
"p-queue": "^8.0.1",
|
"p-queue": "^8.0.1",
|
||||||
"path-browserify": "^1.0.1",
|
"path-browserify": "^1.0.1",
|
||||||
"process": "^0.11.10",
|
"process": "^0.11.10",
|
||||||
|
|||||||
104
pro/LICENSE
Normal file
104
pro/LICENSE
Normal file
@ -0,0 +1,104 @@
|
|||||||
|
# PolyForm Strict License 1.0.0
|
||||||
|
|
||||||
|
<https://polyformproject.org/licenses/strict/1.0.0>
|
||||||
|
|
||||||
|
## Acceptance
|
||||||
|
|
||||||
|
In order to get any license under these terms, you must agree
|
||||||
|
to them as both strict obligations and conditions to all
|
||||||
|
your licenses.
|
||||||
|
|
||||||
|
## Copyright License
|
||||||
|
|
||||||
|
The licensor grants you a copyright license for the software
|
||||||
|
to do everything you might do with the software that would
|
||||||
|
otherwise infringe the licensor's copyright in it for any
|
||||||
|
permitted purpose, other than distributing the software or
|
||||||
|
making changes or new works based on the software.
|
||||||
|
|
||||||
|
## Patent License
|
||||||
|
|
||||||
|
The licensor grants you a patent license for the software that
|
||||||
|
covers patent claims the licensor can license, or becomes able
|
||||||
|
to license, that you would infringe by using the software.
|
||||||
|
|
||||||
|
## Noncommercial Purposes
|
||||||
|
|
||||||
|
Any noncommercial purpose is a permitted purpose.
|
||||||
|
|
||||||
|
## Personal Uses
|
||||||
|
|
||||||
|
Personal use for research, experiment, and testing for
|
||||||
|
the benefit of public knowledge, personal study, private
|
||||||
|
entertainment, hobby projects, amateur pursuits, or religious
|
||||||
|
observance, without any anticipated commercial application,
|
||||||
|
is use for a permitted purpose.
|
||||||
|
|
||||||
|
## Noncommercial Organizations
|
||||||
|
|
||||||
|
Use by any charitable organization, educational institution,
|
||||||
|
public research organization, public safety or health
|
||||||
|
organization, environmental protection organization,
|
||||||
|
or government institution is use for a permitted purpose
|
||||||
|
regardless of the source of funding or obligations resulting
|
||||||
|
from the funding.
|
||||||
|
|
||||||
|
## Fair Use
|
||||||
|
|
||||||
|
You may have "fair use" rights for the software under the
|
||||||
|
law. These terms do not limit them.
|
||||||
|
|
||||||
|
## No Other Rights
|
||||||
|
|
||||||
|
These terms do not allow you to sublicense or transfer any of
|
||||||
|
your licenses to anyone else, or prevent the licensor from
|
||||||
|
granting licenses to anyone else. These terms do not imply
|
||||||
|
any other licenses.
|
||||||
|
|
||||||
|
## Patent Defense
|
||||||
|
|
||||||
|
If you make any written claim that the software infringes or
|
||||||
|
contributes to infringement of any patent, your patent license
|
||||||
|
for the software granted under these terms ends immediately. If
|
||||||
|
your company makes such a claim, your patent license ends
|
||||||
|
immediately for work on behalf of your company.
|
||||||
|
|
||||||
|
## Violations
|
||||||
|
|
||||||
|
The first time you are notified in writing that you have
|
||||||
|
violated any of these terms, or done anything with the software
|
||||||
|
not covered by your licenses, your licenses can nonetheless
|
||||||
|
continue if you come into full compliance with these terms,
|
||||||
|
and take practical steps to correct past violations, within
|
||||||
|
32 days of receiving notice. Otherwise, all your licenses
|
||||||
|
end immediately.
|
||||||
|
|
||||||
|
## No Liability
|
||||||
|
|
||||||
|
***As far as the law allows, the software comes as is, without
|
||||||
|
any warranty or condition, and the licensor will not be liable
|
||||||
|
to you for any damages arising out of these terms or the use
|
||||||
|
or nature of the software, under any kind of legal claim.***
|
||||||
|
|
||||||
|
## Definitions
|
||||||
|
|
||||||
|
The **licensor** is the individual or entity offering these
|
||||||
|
terms, and the **software** is the software the licensor makes
|
||||||
|
available under these terms.
|
||||||
|
|
||||||
|
**You** refers to the individual or entity agreeing to these
|
||||||
|
terms.
|
||||||
|
|
||||||
|
**Your company** is any legal entity, sole proprietorship,
|
||||||
|
or other kind of organization that you work for, plus all
|
||||||
|
organizations that have control over, are under the control of,
|
||||||
|
or are under common control with that organization. **Control**
|
||||||
|
means ownership of substantially all the assets of an entity,
|
||||||
|
or the power to direct its management and policies by vote,
|
||||||
|
contract, or otherwise. Control can be direct or indirect.
|
||||||
|
|
||||||
|
**Your licenses** are all the licenses granted to you for the
|
||||||
|
software under these terms.
|
||||||
|
|
||||||
|
**Use** means anything you do with the software requiring one
|
||||||
|
of your licenses.
|
||||||
21
pro/README.md
Normal file
21
pro/README.md
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
# Pro Features
|
||||||
|
|
||||||
|
## What?
|
||||||
|
|
||||||
|
Remotely Save has some "pro features", which users have to pay for using them.
|
||||||
|
|
||||||
|
## Sign Up / Sign In
|
||||||
|
|
||||||
|
Please go to <https://remotelysave.com> to sign up and sign in an account firstly.
|
||||||
|
|
||||||
|
## Smart Conflict
|
||||||
|
|
||||||
|
Basic (free) version can detect conflicts, but users have to choose to keep newer version or larger version of the files.
|
||||||
|
|
||||||
|
PRO (paid) feature "Smart Conflict" gives users one more option: merge small markdown files, or duplicate large markdown files or non-markdown files.
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
The codes or files or subfolders inside the current folder (`pro` in the repo), are released under "source available" license: "PolyForm Strict License 1.0.0".
|
||||||
|
|
||||||
|
Suggestions are welcome.
|
||||||
302
pro/src/account.ts
Normal file
302
pro/src/account.ts
Normal file
@ -0,0 +1,302 @@
|
|||||||
|
import { nanoid } from "nanoid";
|
||||||
|
import { base64url } from "rfc4648";
|
||||||
|
import {
|
||||||
|
OAUTH2_FORCE_EXPIRE_MILLISECONDS,
|
||||||
|
type RemotelySavePluginSettings,
|
||||||
|
} from "../../src/baseTypes";
|
||||||
|
import {
|
||||||
|
COMMAND_CALLBACK_PRO,
|
||||||
|
type FeatureInfo,
|
||||||
|
PRO_CLIENT_ID,
|
||||||
|
type PRO_FEATURE_TYPE,
|
||||||
|
PRO_WEBSITE,
|
||||||
|
type ProConfig,
|
||||||
|
} from "./baseTypesPro";
|
||||||
|
|
||||||
|
const site = PRO_WEBSITE;
|
||||||
|
console.debug(`remotelysave official website: ${site}`);
|
||||||
|
|
||||||
|
export const DEFAULT_PRO_CONFIG: ProConfig = {
|
||||||
|
accessToken: "",
|
||||||
|
accessTokenExpiresInMs: 0,
|
||||||
|
accessTokenExpiresAtTimeMs: 0,
|
||||||
|
refreshToken: "",
|
||||||
|
enabledProFeatures: [],
|
||||||
|
email: "",
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* https://datatracker.ietf.org/doc/html/rfc7636
|
||||||
|
* dBjftJeZ4CVP-mB92K27uhbUJU1p1r_wW1gFWFOEjXk
|
||||||
|
* => E9Melhoa2OwvFrEMTJguCHaoeK1t8URWbuGJSstw-cM
|
||||||
|
* @param x
|
||||||
|
* @returns BASE64URL-ENCODE(SHA256(ASCII(code_verifier)))
|
||||||
|
*/
|
||||||
|
async function codeVerifier2CodeChallenge(x: string) {
|
||||||
|
if (x === undefined || x === "") {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
return base64url.stringify(
|
||||||
|
new Uint8Array(
|
||||||
|
await crypto.subtle.digest("SHA-256", new TextEncoder().encode(x))
|
||||||
|
),
|
||||||
|
{
|
||||||
|
pad: false,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
} catch (e) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export const generateAuthUrlAndCodeVerifierChallenge = async (
|
||||||
|
hasCallback: boolean
|
||||||
|
) => {
|
||||||
|
const appKey = PRO_CLIENT_ID ?? "cli-"; // hard-code
|
||||||
|
const codeVerifier = nanoid(128);
|
||||||
|
const codeChallenge = await codeVerifier2CodeChallenge(codeVerifier);
|
||||||
|
let authUrl = `${site}/oauth2/authorize?response_type=code&client_id=${appKey}&token_access_type=offline&code_challenge_method=S256&code_challenge=${codeChallenge}&scope=pro.list.read`;
|
||||||
|
if (hasCallback) {
|
||||||
|
authUrl += `&redirect_uri=obsidian://${COMMAND_CALLBACK_PRO}`;
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
authUrl,
|
||||||
|
codeVerifier,
|
||||||
|
codeChallenge,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export const sendAuthReq = async (
|
||||||
|
verifier: string,
|
||||||
|
authCode: string,
|
||||||
|
errorCallBack: any
|
||||||
|
) => {
|
||||||
|
const appKey = PRO_CLIENT_ID ?? "cli-"; // hard-code
|
||||||
|
try {
|
||||||
|
const k = {
|
||||||
|
code: authCode,
|
||||||
|
grant_type: "authorization_code",
|
||||||
|
code_verifier: verifier,
|
||||||
|
client_id: appKey,
|
||||||
|
// redirect_uri: `obsidian://${COMMAND_CALLBACK_PRO}`,
|
||||||
|
scope: "pro.list.read",
|
||||||
|
};
|
||||||
|
// console.debug(k);
|
||||||
|
const resp1 = await fetch(`${site}/api/v1/oauth2/token`, {
|
||||||
|
method: "POST",
|
||||||
|
body: new URLSearchParams(k),
|
||||||
|
});
|
||||||
|
const resp2 = await resp1.json();
|
||||||
|
return resp2;
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e);
|
||||||
|
if (errorCallBack !== undefined) {
|
||||||
|
await errorCallBack(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const sendRefreshTokenReq = async (refreshToken: string) => {
|
||||||
|
const appKey = PRO_CLIENT_ID ?? "cli-"; // hard-code
|
||||||
|
try {
|
||||||
|
console.info("start auto getting refreshed Remotely Save access token.");
|
||||||
|
const resp1 = await fetch(`${site}/api/v1/oauth2/token`, {
|
||||||
|
method: "POST",
|
||||||
|
body: new URLSearchParams({
|
||||||
|
grant_type: "refresh_token",
|
||||||
|
refresh_token: refreshToken,
|
||||||
|
client_id: appKey,
|
||||||
|
scope: "pro.list.read",
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
const resp2: AuthResError | AuthResSucc = await resp1.json();
|
||||||
|
console.info("finish auto getting refreshed Remotely Save access token.");
|
||||||
|
return resp2;
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e);
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
interface AuthResError {
|
||||||
|
error: "invalid_request";
|
||||||
|
}
|
||||||
|
|
||||||
|
interface AuthResSucc {
|
||||||
|
error: undefined; // needed for typescript
|
||||||
|
refresh_token?: string;
|
||||||
|
access_token: string;
|
||||||
|
expires_in: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const setConfigBySuccessfullAuthInplace = async (
|
||||||
|
config: ProConfig,
|
||||||
|
authRes: AuthResError | AuthResSucc,
|
||||||
|
saveUpdatedConfigFunc: () => Promise<any> | undefined
|
||||||
|
) => {
|
||||||
|
if (authRes.error !== undefined) {
|
||||||
|
throw Error(`you should not save the setting for ${authRes.error}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
config.accessToken = authRes.access_token;
|
||||||
|
config.accessTokenExpiresAtTimeMs =
|
||||||
|
Date.now() + authRes.expires_in * 1000 - 5 * 60 * 1000;
|
||||||
|
config.accessTokenExpiresInMs = authRes.expires_in * 1000;
|
||||||
|
config.refreshToken = authRes.refresh_token || config.refreshToken;
|
||||||
|
|
||||||
|
// manually set it expired after 80 days;
|
||||||
|
config.credentialsShouldBeDeletedAtTimeMs =
|
||||||
|
Date.now() + OAUTH2_FORCE_EXPIRE_MILLISECONDS;
|
||||||
|
|
||||||
|
await saveUpdatedConfigFunc?.();
|
||||||
|
|
||||||
|
console.info(
|
||||||
|
"finish updating local info of Remotely Save official website token"
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getAccessToken = async (
|
||||||
|
config: ProConfig,
|
||||||
|
saveUpdatedConfigFunc: () => Promise<any> | undefined
|
||||||
|
) => {
|
||||||
|
const ts = Date.now();
|
||||||
|
if (
|
||||||
|
config.accessToken !== undefined &&
|
||||||
|
config.accessToken !== "" &&
|
||||||
|
config.accessTokenExpiresAtTimeMs > ts &&
|
||||||
|
(config.credentialsShouldBeDeletedAtTimeMs ?? ts + 1000 * 1000) > ts
|
||||||
|
) {
|
||||||
|
return config.accessToken;
|
||||||
|
}
|
||||||
|
|
||||||
|
console.debug(
|
||||||
|
`currently, accessToken=${config.accessToken}, accessTokenExpiresAtTimeMs=${
|
||||||
|
config.accessTokenExpiresAtTimeMs
|
||||||
|
}, credentialsShouldBeDeletedAtTimeMs=${
|
||||||
|
config.credentialsShouldBeDeletedAtTimeMs
|
||||||
|
},comp1=${config.accessTokenExpiresAtTimeMs > ts}, comp2=${
|
||||||
|
(config.credentialsShouldBeDeletedAtTimeMs ?? ts + 1000 * 1000) > ts
|
||||||
|
}`
|
||||||
|
);
|
||||||
|
|
||||||
|
// try to get it again??
|
||||||
|
const res = await sendRefreshTokenReq(config.refreshToken ?? "refresh-");
|
||||||
|
await setConfigBySuccessfullAuthInplace(config, res, saveUpdatedConfigFunc);
|
||||||
|
|
||||||
|
if (res.error !== undefined) {
|
||||||
|
throw Error("cannot update accessToken");
|
||||||
|
}
|
||||||
|
return res.access_token;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getAndSaveProFeatures = async (
|
||||||
|
config: ProConfig,
|
||||||
|
pluginVersion: string,
|
||||||
|
saveUpdatedConfigFunc: () => Promise<any> | undefined
|
||||||
|
) => {
|
||||||
|
const access = await getAccessToken(config, saveUpdatedConfigFunc);
|
||||||
|
|
||||||
|
const resp1 = await fetch(`${site}/api/v1/pro/list`, {
|
||||||
|
method: "GET",
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${access}`,
|
||||||
|
"REMOTELYSAVE-API-Plugin-Ver": pluginVersion,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
const rsp2: {
|
||||||
|
proFeatures: FeatureInfo[];
|
||||||
|
} = await resp1.json();
|
||||||
|
|
||||||
|
config.enabledProFeatures = rsp2.proFeatures;
|
||||||
|
await saveUpdatedConfigFunc?.();
|
||||||
|
return rsp2;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getAndSaveProEmail = async (
|
||||||
|
config: ProConfig,
|
||||||
|
pluginVersion: string,
|
||||||
|
saveUpdatedConfigFunc: () => Promise<any> | undefined
|
||||||
|
) => {
|
||||||
|
const access = await getAccessToken(config, saveUpdatedConfigFunc);
|
||||||
|
|
||||||
|
const resp1 = await fetch(`${site}/api/v1/profile/list`, {
|
||||||
|
method: "GET",
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${access}`,
|
||||||
|
"REMOTELYSAVE-API-Plugin-Ver": pluginVersion,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
const rsp2: {
|
||||||
|
email: string;
|
||||||
|
} = await resp1.json();
|
||||||
|
|
||||||
|
config.email = rsp2.email;
|
||||||
|
await saveUpdatedConfigFunc?.();
|
||||||
|
return rsp2;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If the check doesn't pass, the function should throw the error
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
export const checkProRunnableAndFixInplace = async (
|
||||||
|
featuresToCheck: PRO_FEATURE_TYPE[],
|
||||||
|
config: RemotelySavePluginSettings,
|
||||||
|
pluginVersion: string,
|
||||||
|
saveUpdatedConfigFunc: () => Promise<any> | undefined
|
||||||
|
): Promise<true> => {
|
||||||
|
// if no pro features are used, we are good to go, no checking
|
||||||
|
if (
|
||||||
|
featuresToCheck.contains("feature-smart_conflict") &&
|
||||||
|
config.conflictAction !== "smart_conflict"
|
||||||
|
) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// many checks if status is valid
|
||||||
|
|
||||||
|
// no account
|
||||||
|
if (config.pro === undefined || config.pro.refreshToken === undefined) {
|
||||||
|
throw Error(`you need to "connect" to your account to use PRO features`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// every features should have at most 40 days expiration dates
|
||||||
|
// and if the time has expired, we also check
|
||||||
|
const msIn40Days = 1000 * 60 * 60 * 24 * 40;
|
||||||
|
for (const f of config.pro.enabledProFeatures) {
|
||||||
|
const tooFarInTheFuture = f.expireAtTimeMs >= Date.now() + msIn40Days;
|
||||||
|
const alreadyExpired = f.expireAtTimeMs <= Date.now();
|
||||||
|
if (tooFarInTheFuture || alreadyExpired) {
|
||||||
|
console.info(
|
||||||
|
`the pro feature is too far in the future and has expired, check again.`
|
||||||
|
);
|
||||||
|
await getAndSaveProFeatures(
|
||||||
|
config.pro,
|
||||||
|
pluginVersion,
|
||||||
|
saveUpdatedConfigFunc
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// check for the features
|
||||||
|
if (featuresToCheck.contains("feature-smart_conflict")) {
|
||||||
|
if (config.conflictAction === "smart_conflict") {
|
||||||
|
if (
|
||||||
|
config.pro.enabledProFeatures.filter(
|
||||||
|
(x) => x.featureName === "feature-smart_conflict"
|
||||||
|
).length === 1
|
||||||
|
) {
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
throw Error(
|
||||||
|
`You're trying to use "smart conflict" PRO feature but you haven't subscribe to it.`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
};
|
||||||
25
pro/src/baseTypesPro.ts
Normal file
25
pro/src/baseTypesPro.ts
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
export const MERGABLE_SIZE = 1000 * 1000; // 1 MB
|
||||||
|
|
||||||
|
export const COMMAND_CALLBACK_PRO = "remotely-save-cb-pro";
|
||||||
|
export const PRO_CLIENT_ID = process.env.DEFAULT_REMOTELYSAVE_CLIENT_ID;
|
||||||
|
export const PRO_WEBSITE = process.env.DEFAULT_REMOTELYSAVE_WEBSITE;
|
||||||
|
|
||||||
|
export type PRO_FEATURE_TYPE =
|
||||||
|
| "feature-smart_conflict"
|
||||||
|
| "feature-google_drive";
|
||||||
|
|
||||||
|
export interface FeatureInfo {
|
||||||
|
featureName: PRO_FEATURE_TYPE;
|
||||||
|
enableAtTimeMs: bigint;
|
||||||
|
expireAtTimeMs: bigint;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ProConfig {
|
||||||
|
email?: string;
|
||||||
|
refreshToken?: string;
|
||||||
|
accessToken: string;
|
||||||
|
accessTokenExpiresInMs: number;
|
||||||
|
accessTokenExpiresAtTimeMs: number;
|
||||||
|
enabledProFeatures: FeatureInfo[];
|
||||||
|
credentialsShouldBeDeletedAtTimeMs?: number;
|
||||||
|
}
|
||||||
257
pro/src/conflictLogic.ts
Normal file
257
pro/src/conflictLogic.ts
Normal file
@ -0,0 +1,257 @@
|
|||||||
|
import isEqual from "lodash/isEqual";
|
||||||
|
// import {
|
||||||
|
// makePatches,
|
||||||
|
// applyPatches,
|
||||||
|
// stringifyPatches,
|
||||||
|
// parsePatch,
|
||||||
|
// } from "@sanity/diff-match-patch";
|
||||||
|
import {
|
||||||
|
LCS,
|
||||||
|
diff3Merge,
|
||||||
|
diffComm,
|
||||||
|
diffPatch,
|
||||||
|
mergeDiff3,
|
||||||
|
mergeDigIn,
|
||||||
|
patch,
|
||||||
|
} from "node-diff3";
|
||||||
|
import type { Entity } from "../../src/baseTypes";
|
||||||
|
import { copyFile } from "../../src/copyLogic";
|
||||||
|
import type { FakeFs } from "../../src/fsAll";
|
||||||
|
import { MERGABLE_SIZE } from "./baseTypesPro";
|
||||||
|
|
||||||
|
export function isMergable(a: Entity, b?: Entity) {
|
||||||
|
if (b !== undefined && a.keyRaw !== b.keyRaw) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
!a.keyRaw.endsWith("/") &&
|
||||||
|
a.sizeRaw <= MERGABLE_SIZE &&
|
||||||
|
(a.keyRaw.endsWith(".md") || a.keyRaw.endsWith(".markdown"))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* slightly modify to adjust in markdown context
|
||||||
|
* @param a
|
||||||
|
* @param o
|
||||||
|
* @param b
|
||||||
|
*/
|
||||||
|
function mergeDigInModified(a: string, o: string, b: string) {
|
||||||
|
const { conflict, result } = mergeDigIn(a, o, b);
|
||||||
|
for (let index = 0; index < result.length; ++index) {
|
||||||
|
if (["<<<<<<<", "=======", ">>>>>>>"].contains(result[index])) {
|
||||||
|
result[index] = "`" + result[index] + "`";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
conflict,
|
||||||
|
result,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function getLCSText(a: string, b: string) {
|
||||||
|
const aa = a.split("\n");
|
||||||
|
const bb = b.split("\n");
|
||||||
|
let raw = LCS(aa, bb);
|
||||||
|
|
||||||
|
const k: string[] = [];
|
||||||
|
|
||||||
|
do {
|
||||||
|
k.unshift(aa[raw.buffer1index]);
|
||||||
|
|
||||||
|
raw = raw.chain as any;
|
||||||
|
} while (raw !== null && raw !== undefined && raw.buffer1index !== -1);
|
||||||
|
|
||||||
|
return k.join("\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* It's tricky. We find LCS then pretend it's the original text
|
||||||
|
* @param a
|
||||||
|
* @param b
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
function twoWayMerge(a: string, b: string): string {
|
||||||
|
// const c = getLCSText(a, b);
|
||||||
|
// const patches = makePatches(c, a);
|
||||||
|
// const [d] = applyPatches(patches, b);
|
||||||
|
const c = getLCSText(a, b);
|
||||||
|
const d = mergeDigInModified(a, c, b).result.join("\n");
|
||||||
|
return d;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Originally three way merge.
|
||||||
|
* @param a
|
||||||
|
* @param b
|
||||||
|
* @param orig
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
function threeWayMerge(a: string, b: string, orig: string) {
|
||||||
|
return mergeDigInModified(a, orig, b).result.join("\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function mergeFile(
|
||||||
|
key: string,
|
||||||
|
left: FakeFs,
|
||||||
|
right: FakeFs,
|
||||||
|
contentOrig: ArrayBuffer | null | undefined
|
||||||
|
) {
|
||||||
|
// console.debug(
|
||||||
|
// `mergeFile: key=${key}, left=${left.kind}, right=${right.kind}`
|
||||||
|
// );
|
||||||
|
if (key.endsWith("/")) {
|
||||||
|
throw Error(`should not call ${key} in mergeFile`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!key.endsWith(".md") && !key.endsWith(".markdown")) {
|
||||||
|
throw Error(`currently only support markdown files in mergeFile`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const [contentLeft, contentRight] = await Promise.all([
|
||||||
|
left.readFile(key),
|
||||||
|
right.readFile(key),
|
||||||
|
]);
|
||||||
|
|
||||||
|
let newArrayBuffer: ArrayBuffer | undefined = undefined;
|
||||||
|
const decoder = new TextDecoder("utf-8");
|
||||||
|
|
||||||
|
if (isEqual(contentLeft, contentRight)) {
|
||||||
|
// we are lucky enough
|
||||||
|
newArrayBuffer = contentLeft;
|
||||||
|
// TODO: save the write
|
||||||
|
} else {
|
||||||
|
if (contentOrig === null || contentOrig === undefined) {
|
||||||
|
const newText = twoWayMerge(
|
||||||
|
decoder.decode(contentLeft),
|
||||||
|
decoder.decode(contentRight)
|
||||||
|
);
|
||||||
|
// no need to worry about the offset here because the array is new and not sliced
|
||||||
|
newArrayBuffer = new TextEncoder().encode(newText).buffer;
|
||||||
|
} else {
|
||||||
|
const newText = threeWayMerge(
|
||||||
|
decoder.decode(contentLeft),
|
||||||
|
decoder.decode(contentRight),
|
||||||
|
decoder.decode(contentOrig)
|
||||||
|
);
|
||||||
|
newArrayBuffer = new TextEncoder().encode(newText).buffer;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const mtime = Date.now();
|
||||||
|
|
||||||
|
// left (local) must wait for the right
|
||||||
|
// because the mtime might be different after upload
|
||||||
|
// upload firstly
|
||||||
|
const rightEntity = await right.writeFile(key, newArrayBuffer, mtime, mtime);
|
||||||
|
// write local secondly
|
||||||
|
const leftEntity = await left.writeFile(
|
||||||
|
key,
|
||||||
|
newArrayBuffer,
|
||||||
|
rightEntity.mtimeCli ?? mtime,
|
||||||
|
rightEntity.mtimeCli ?? mtime
|
||||||
|
);
|
||||||
|
|
||||||
|
return {
|
||||||
|
entity: rightEntity,
|
||||||
|
content: newArrayBuffer,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getFileRename(key: string) {
|
||||||
|
if (
|
||||||
|
key === "" ||
|
||||||
|
key === "." ||
|
||||||
|
key === ".." ||
|
||||||
|
key === "/" ||
|
||||||
|
key.endsWith("/")
|
||||||
|
) {
|
||||||
|
throw Error(`we cannot rename key=${key}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const segsPath = key.split("/");
|
||||||
|
const name = segsPath[segsPath.length - 1];
|
||||||
|
const segsName = name.split(".");
|
||||||
|
|
||||||
|
if (segsName.length === 0) {
|
||||||
|
throw Error(`we cannot rename key=${key}`);
|
||||||
|
} else if (segsName.length === 1) {
|
||||||
|
// name = "kkk" without any dot
|
||||||
|
segsPath[segsPath.length - 1] = `${name}.dup`;
|
||||||
|
} else if (segsName.length === 2) {
|
||||||
|
if (segsName[0] === "") {
|
||||||
|
// name = ".kkkk" with leading dot
|
||||||
|
segsPath[segsPath.length - 1] = `${name}.dup`;
|
||||||
|
} else if (segsName[1] === "") {
|
||||||
|
// name = "kkkk." with tailing dot
|
||||||
|
segsPath[segsPath.length - 1] = `${segsName[0]}.dup`;
|
||||||
|
} else {
|
||||||
|
// name = "aaa.bbb" normally
|
||||||
|
segsPath[segsPath.length - 1] = `${segsName[0]}.dup.${segsName[1]}`;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// name = "[...].bbb.ccc"
|
||||||
|
const firstPart = segsName.slice(0, segsName.length - 1).join(".");
|
||||||
|
const thirdPart = segsName[segsName.length - 1];
|
||||||
|
segsPath[segsPath.length - 1] = `${firstPart}.dup.${thirdPart}`;
|
||||||
|
}
|
||||||
|
const res = segsPath.join("/");
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* local: x.md -> x.dup.md -> upload to remote
|
||||||
|
* remote: x.md -> download to local -> using original name x.md
|
||||||
|
*/
|
||||||
|
export async function duplicateFile(
|
||||||
|
key: string,
|
||||||
|
left: FakeFs,
|
||||||
|
right: FakeFs,
|
||||||
|
uploadCallback: (entity: Entity) => Promise<any>,
|
||||||
|
downloadCallback: (entity: Entity) => Promise<any>
|
||||||
|
) {
|
||||||
|
let key2 = getFileRename(key);
|
||||||
|
let usable = false;
|
||||||
|
do {
|
||||||
|
try {
|
||||||
|
const s = await left.stat(key2);
|
||||||
|
if (s === null || s === undefined) {
|
||||||
|
throw Error(`not exist $${key2}`);
|
||||||
|
}
|
||||||
|
console.debug(`key2=${key2} exists, cannot use for new file`);
|
||||||
|
key2 = getFileRename(key2);
|
||||||
|
console.debug(`key2=${key2} is prepared for next try`);
|
||||||
|
} catch (e) {
|
||||||
|
// not exists, exactly what we want
|
||||||
|
console.debug(`key2=${key2} doesn't exist, usable for new file`);
|
||||||
|
usable = true;
|
||||||
|
}
|
||||||
|
} while (!usable);
|
||||||
|
await left.rename(key, key2);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* x.dup.md -> upload to remote
|
||||||
|
*/
|
||||||
|
async function f1() {
|
||||||
|
const k = await copyFile(key2, left, right);
|
||||||
|
await uploadCallback(k.entity);
|
||||||
|
return k.entity;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* x.md -> download to local
|
||||||
|
*/
|
||||||
|
async function f2() {
|
||||||
|
const k = await copyFile(key, right, left);
|
||||||
|
await downloadCallback(k.entity);
|
||||||
|
return k.entity;
|
||||||
|
}
|
||||||
|
|
||||||
|
const [resUpload, resDownload] = await Promise.all([f1(), f2()]);
|
||||||
|
|
||||||
|
return {
|
||||||
|
upload: resUpload,
|
||||||
|
download: resDownload,
|
||||||
|
};
|
||||||
|
}
|
||||||
39
pro/src/langs/en.json
Normal file
39
pro/src/langs/en.json
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
{
|
||||||
|
"settings_conflictaction_smart_conflict": "Smart Conflict (PRO) (beta)",
|
||||||
|
"settings_conflictaction_smart_conflict_desc": "<p><strong>!!It's a PRO feature! You need an online account for this feature!!</strong>(<a href=\"#settings-pro\">scroll down</a> for more info about PRO account.)</p><p><ul><li>For small markdown files, the plugin tries to merge them with diff3 algorithm.</li><li>For large files or not-markdown files, the plugin saves both files by renaming them.</li></ul></p><p><strong>Please manually backup your vaule before using this feature!</strong></p>",
|
||||||
|
|
||||||
|
"protocol_pro_connecting": "Connectting",
|
||||||
|
"protocol_pro_connect_manualinput_succ": "You've connected",
|
||||||
|
"protocol_pro_connect_fail": "Something went wrong from response from Remotely Save official website. Maybe the network connection is not good. Maybe you rejected the auth?",
|
||||||
|
"protocol_pro_connect_succ_revoke": "You've connected as user {{email}}. If you want to disconnect, click this button.",
|
||||||
|
|
||||||
|
"modal_prorevokeauth": "Revoke auth by clicking here and follow the steps.",
|
||||||
|
"modal_prorevokeauth_clean": "Clean",
|
||||||
|
"modal_prorevokeauth_clean_desc": "Clean local auth record",
|
||||||
|
"modal_prorevokeauth_clean_button": "Clean",
|
||||||
|
"modal_prorevokeauth_clean_notice": "Local auth record is cleaned",
|
||||||
|
"modal_prorevokeauth_clean_fail": "Fail to clean local auth record.",
|
||||||
|
"modal_proauth_copybutton": "Click to copy the auth url",
|
||||||
|
"modal_proauth_copynotice": "The auth url is copied to the clipboard!",
|
||||||
|
"modal_proauth_maualinput": "The Code from the website",
|
||||||
|
"modal_proauth_maualinput_desc": "Please input the code here from the end of auth flow, and press confirm.",
|
||||||
|
"modal_proauth_maualinput_notice": "Trying to connect, wait...",
|
||||||
|
"modal_proauth_maualinput_conn_fail": "Failed to connect",
|
||||||
|
|
||||||
|
"settings_pro": "Account (for PRO features)",
|
||||||
|
"settings_pro_tutorial": "<p>Using <stong>basic</strong> features of Remotely Save is <strong>FREE</strong> and do <strong>NOT</strong> need an account.</p><p>However, you will <strong>need</strong> an online account and <strong>PAY</strong> for the <strong>PRO</strong> features such as smart conflict.</p><p>Firstly please click the button to sign up and sign in to the website: <a href=\"https://remotelysave.com\">https://remotelysave.com</a>. Notice: It's different from, and NOT affiliated with Obsidian account.</p><p>Secondly please \"connect\" your local device to your online account.",
|
||||||
|
"settings_pro_features": "Features",
|
||||||
|
"settings_pro_features_desc": "Here are features you've enabled:<br/>{{{features}}}",
|
||||||
|
"settings_pro_features_refresh_button": "Check again",
|
||||||
|
"settings_pro_features_refresh_fetch": "Fetching...",
|
||||||
|
"settings_pro_features_refresh_succ": "Refreshed!",
|
||||||
|
"settings_pro_revoke": "Disconnect",
|
||||||
|
"settings_pro_revoke_desc": "You've connected as user {{email}}. If you want to disconnect, click this button.",
|
||||||
|
"settings_pro_revoke_button": "Disconnect",
|
||||||
|
"settings_pro_intro": "Remotely Save Online Account",
|
||||||
|
"settings_pro_intro_desc": "Click the button to jump to the website to sign up or sign in.",
|
||||||
|
"settings_pro_intro_button": "Sign Up / Sign In",
|
||||||
|
"settings_pro_auth": "Connect",
|
||||||
|
"settings_pro_auth_desc": "After you sign up and sign in the account on the website, you need to connect your plugin here to the online account. Please click the button to connect.",
|
||||||
|
"settings_pro_auth_button": "Connect"
|
||||||
|
}
|
||||||
9
pro/src/langs/index.ts
Normal file
9
pro/src/langs/index.ts
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
import en from "./en.json";
|
||||||
|
import zh_cn from "./zh_cn.json";
|
||||||
|
import zh_tw from "./zh_tw.json";
|
||||||
|
|
||||||
|
export const LANGS = {
|
||||||
|
en: en,
|
||||||
|
zh_cn: zh_cn,
|
||||||
|
zh_tw: zh_tw,
|
||||||
|
};
|
||||||
39
pro/src/langs/zh_cn.json
Normal file
39
pro/src/langs/zh_cn.json
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
{
|
||||||
|
"settings_conflictaction_smart_conflict": "智能处理冲突 (PRO) (beta)",
|
||||||
|
"settings_conflictaction_smart_conflict_desc": "<p><strong>!!这是 PRO(付费)功能! 您需要在线账号来使用此功能!!</strong>(<a href=\"#settings-pro\">向下滑</a>可以看到 PRO 账号的更多信息。)</p><p><ul><li>小 markdown 文件,本插件尝试使用 diff3 算法合并它;</li><li>对于大文件或非 markdown 文件,本插件尝试改名字并均进行保存。</li></ul></p><p><strong>请注意先手动备份 vault 文件再用此功能!</strong></p>",
|
||||||
|
|
||||||
|
"protocol_pro_connecting": "正在连接",
|
||||||
|
"protocol_pro_connect_manualinput_succ": "连接成功",
|
||||||
|
"protocol_pro_connect_fail": "Remotely Save 官网返回错误。可能是网络连接不稳定。也可能是您拒绝了授权?",
|
||||||
|
"protocol_pro_connect_succ_revoke": "您已连接上账号 {{email}}。如果要取消连接,请点击此按钮。",
|
||||||
|
|
||||||
|
"modal_prorevokeauth": "点击这里和按照步骤取消授权。",
|
||||||
|
"modal_prorevokeauth_clean": "清理",
|
||||||
|
"modal_prorevokeauth_clean_desc": "清理本地授权记录",
|
||||||
|
"modal_prorevokeauth_clean_button": "清理",
|
||||||
|
"modal_prorevokeauth_clean_notice": "清理本地授权记录完毕",
|
||||||
|
"modal_prorevokeauth_clean_fail": "清理本地授权记录粗错。",
|
||||||
|
"modal_proauth_copybutton": "点击从而复制授权网址",
|
||||||
|
"modal_proauth_copynotice": "授权网址已复制!",
|
||||||
|
"modal_proauth_maualinput": "网站的授权码",
|
||||||
|
"modal_proauth_maualinput_desc": "请输入授权流程最后一步的授权码,然后点击确认。",
|
||||||
|
"modal_proauth_maualinput_notice": "正在连接,请稍候......",
|
||||||
|
"modal_proauth_maualinput_conn_fail": "连接失败",
|
||||||
|
|
||||||
|
"settings_pro": "账号(PRO 付费功能)",
|
||||||
|
"settings_pro_tutorial": "<p>使用 Remotely Save 的<stong>基本</strong>功能是<strong>免费的</strong>,而且<strong>不</strong>需要注册对应账号。</p><p>但是,您<strong>需要</strong>注册账号和对<strong>PRO</strong>功能<strong>付费</strong>使用,如智能处理冲突功能。</p><p>第一步:点击按钮从而注册和登录网站:<a href=\"https://remotelysave.com\">https://remotelysave.com</a>。注意:这和 Obsidian 官方账号无关,是不同的账号。</p><p>第二部:点击“连接”按钮,从而连接本设备和在线账号。",
|
||||||
|
"settings_pro_features": "功能",
|
||||||
|
"settings_pro_features_desc": "您开通了以下功能:<br/>{{{features}}}",
|
||||||
|
"settings_pro_features_refresh_button": "再次检查",
|
||||||
|
"settings_pro_features_refresh_fetch": "正在获取数据......",
|
||||||
|
"settings_pro_features_refresh_succ": "已刷新!",
|
||||||
|
"settings_pro_revoke": "断开连接",
|
||||||
|
"settings_pro_revoke_desc": "您已连接上账号 {{email}}。如果要取消连接,请点击此按钮。",
|
||||||
|
"settings_pro_revoke_button": "断开连接",
|
||||||
|
"settings_pro_intro": "Remotely Save 账号",
|
||||||
|
"settings_pro_intro_desc": "点击此按钮,从而到网站上注册和登录。",
|
||||||
|
"settings_pro_intro_button": "注册或登录",
|
||||||
|
"settings_pro_auth": "连接",
|
||||||
|
"settings_pro_auth_desc": "在网站上注册和登录后,您需要“连接”本设备和在线账号。请点击按钮开始连接。",
|
||||||
|
"settings_pro_auth_button": "连接"
|
||||||
|
}
|
||||||
39
pro/src/langs/zh_tw.json
Normal file
39
pro/src/langs/zh_tw.json
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
{
|
||||||
|
"settings_conflictaction_smart_conflict": "智慧處理衝突 (PRO) (beta)",
|
||||||
|
"settings_conflictaction_smart_conflict_desc": "<p><strong>!!這是 PRO(付費)功能! 您需要線上賬號來使用此功能!!</strong>(<a href=\"#settings-pro\">向下滑</a>可以看到 PRO 賬號的更多資訊。)</p><p><ul><li>小 markdown 檔案,本外掛嘗試使用 diff3 演算法合併它;</li><li>對於大檔案或非 markdown 檔案,本外掛嘗試改名字並均進行儲存。</li></ul></p><p><strong>請注意先手動備份 vault 檔案再用此功能!</strong></p>",
|
||||||
|
|
||||||
|
"protocol_pro_connecting": "正在連線",
|
||||||
|
"protocol_pro_connect_manualinput_succ": "連線成功",
|
||||||
|
"protocol_pro_connect_fail": "Remotely Save 官網返回錯誤。可能是網路連線不穩定。也可能是您拒絕了授權?",
|
||||||
|
"protocol_pro_connect_succ_revoke": "您已連線上賬號 {{email}}。如果要取消連線,請點選此按鈕。",
|
||||||
|
|
||||||
|
"modal_prorevokeauth": "點選這裡和按照步驟取消授權。",
|
||||||
|
"modal_prorevokeauth_clean": "清理",
|
||||||
|
"modal_prorevokeauth_clean_desc": "清理本地授權記錄",
|
||||||
|
"modal_prorevokeauth_clean_button": "清理",
|
||||||
|
"modal_prorevokeauth_clean_notice": "清理本地授權記錄完畢",
|
||||||
|
"modal_prorevokeauth_clean_fail": "清理本地授權記錄粗錯。",
|
||||||
|
"modal_proauth_copybutton": "點選從而複製授權網址",
|
||||||
|
"modal_proauth_copynotice": "授權網址已複製!",
|
||||||
|
"modal_proauth_maualinput": "網站的授權碼",
|
||||||
|
"modal_proauth_maualinput_desc": "請輸入授權流程最後一步的授權碼,然後點選確認。",
|
||||||
|
"modal_proauth_maualinput_notice": "正在連線,請稍候......",
|
||||||
|
"modal_proauth_maualinput_conn_fail": "連線失敗",
|
||||||
|
|
||||||
|
"settings_pro": "賬號(PRO 付費功能)",
|
||||||
|
"settings_pro_tutorial": "<p>使用 Remotely Save 的<stong>基本</strong>功能是<strong>免費的</strong>,而且<strong>不</strong>需要註冊對應賬號。</p><p>但是,您<strong>需要</strong>註冊賬號和對<strong>PRO</strong>功能<strong>付費</strong>使用,如智慧處理衝突功能。</p><p>第一步:點選按鈕從而註冊和登入網站:<a href=\"https://remotelysave.com\">https://remotelysave.com</a>。注意:這和 Obsidian 官方賬號無關,是不同的賬號。</p><p>第二部:點選“連線”按鈕,從而連線本裝置和線上賬號。",
|
||||||
|
"settings_pro_features": "功能",
|
||||||
|
"settings_pro_features_desc": "您開通了以下功能:<br/>{{{features}}}",
|
||||||
|
"settings_pro_features_refresh_button": "再次檢查",
|
||||||
|
"settings_pro_features_refresh_fetch": "正在獲取資料......",
|
||||||
|
"settings_pro_features_refresh_succ": "已重新整理!",
|
||||||
|
"settings_pro_revoke": "斷開連線",
|
||||||
|
"settings_pro_revoke_desc": "您已連線上賬號 {{email}}。如果要取消連線,請點選此按鈕。",
|
||||||
|
"settings_pro_revoke_button": "斷開連線",
|
||||||
|
"settings_pro_intro": "Remotely Save 賬號",
|
||||||
|
"settings_pro_intro_desc": "點選此按鈕,從而到網站上註冊和登入。",
|
||||||
|
"settings_pro_intro_button": "註冊或登入",
|
||||||
|
"settings_pro_auth": "連線",
|
||||||
|
"settings_pro_auth_desc": "在網站上註冊和登入後,您需要“連線”本裝置和線上賬號。請點選按鈕開始連線。",
|
||||||
|
"settings_pro_auth_button": "連線"
|
||||||
|
}
|
||||||
47
pro/src/localdb.ts
Normal file
47
pro/src/localdb.ts
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
import type { Entity } from "../../src/baseTypes";
|
||||||
|
import type { InternalDBs } from "../../src/localdb";
|
||||||
|
|
||||||
|
export const upsertFileContentHistoryByVaultAndProfile = async (
|
||||||
|
db: InternalDBs,
|
||||||
|
vaultRandomID: string,
|
||||||
|
profileID: string,
|
||||||
|
prevSync: Entity,
|
||||||
|
prevContent: ArrayBuffer
|
||||||
|
) => {
|
||||||
|
await db.fileContentHistoryTbl.setItem(
|
||||||
|
`${vaultRandomID}\t${profileID}\t${prevSync.key}`,
|
||||||
|
prevContent
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getFileContentHistoryByVaultAndProfile = async (
|
||||||
|
db: InternalDBs,
|
||||||
|
vaultRandomID: string,
|
||||||
|
profileID: string,
|
||||||
|
prevSync: Entity
|
||||||
|
) => {
|
||||||
|
return (await db.fileContentHistoryTbl.getItem(
|
||||||
|
`${vaultRandomID}\t${profileID}\t${prevSync.key}`
|
||||||
|
)) as ArrayBuffer | null | undefined;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const clearFileContentHistoryByVaultAndProfile = async (
|
||||||
|
db: InternalDBs,
|
||||||
|
vaultRandomID: string,
|
||||||
|
profileID: string,
|
||||||
|
key: string
|
||||||
|
) => {
|
||||||
|
await db.fileContentHistoryTbl.removeItem(
|
||||||
|
`${vaultRandomID}\t${profileID}\t${key}`
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const clearAllFileContentHistoryByVault = async (
|
||||||
|
db: InternalDBs,
|
||||||
|
vaultRandomID: string
|
||||||
|
) => {
|
||||||
|
const keys = (await db.fileContentHistoryTbl.keys()).filter((x) =>
|
||||||
|
x.startsWith(`${vaultRandomID}\t`)
|
||||||
|
);
|
||||||
|
await db.fileContentHistoryTbl.removeItems(keys);
|
||||||
|
};
|
||||||
359
pro/src/settingsPro.ts
Normal file
359
pro/src/settingsPro.ts
Normal file
@ -0,0 +1,359 @@
|
|||||||
|
import cloneDeep from "lodash/cloneDeep";
|
||||||
|
import { type App, Modal, Notice, Setting } from "obsidian";
|
||||||
|
import { features } from "process";
|
||||||
|
import type { TransItemType } from "../../src/i18n";
|
||||||
|
import type RemotelySavePlugin from "../../src/main";
|
||||||
|
import { stringToFragment } from "../../src/misc";
|
||||||
|
import {
|
||||||
|
DEFAULT_PRO_CONFIG,
|
||||||
|
generateAuthUrlAndCodeVerifierChallenge,
|
||||||
|
getAndSaveProEmail,
|
||||||
|
getAndSaveProFeatures,
|
||||||
|
sendAuthReq,
|
||||||
|
setConfigBySuccessfullAuthInplace,
|
||||||
|
} from "./account";
|
||||||
|
import {
|
||||||
|
type FeatureInfo,
|
||||||
|
PRO_CLIENT_ID,
|
||||||
|
type ProConfig,
|
||||||
|
} from "./baseTypesPro";
|
||||||
|
|
||||||
|
export class ProAuthModal extends Modal {
|
||||||
|
readonly plugin: RemotelySavePlugin;
|
||||||
|
readonly authDiv: HTMLDivElement;
|
||||||
|
readonly revokeAuthDiv: HTMLDivElement;
|
||||||
|
readonly revokeAuthSetting: Setting;
|
||||||
|
readonly proFeaturesListSetting: Setting;
|
||||||
|
readonly t: (x: TransItemType, vars?: any) => string;
|
||||||
|
constructor(
|
||||||
|
app: App,
|
||||||
|
plugin: RemotelySavePlugin,
|
||||||
|
authDiv: HTMLDivElement,
|
||||||
|
revokeAuthDiv: HTMLDivElement,
|
||||||
|
revokeAuthSetting: Setting,
|
||||||
|
proFeaturesListSetting: Setting,
|
||||||
|
t: (x: TransItemType, vars?: any) => string
|
||||||
|
) {
|
||||||
|
super(app);
|
||||||
|
this.plugin = plugin;
|
||||||
|
this.authDiv = authDiv;
|
||||||
|
this.revokeAuthDiv = revokeAuthDiv;
|
||||||
|
this.revokeAuthSetting = revokeAuthSetting;
|
||||||
|
this.proFeaturesListSetting = proFeaturesListSetting;
|
||||||
|
this.t = t;
|
||||||
|
}
|
||||||
|
|
||||||
|
async onOpen() {
|
||||||
|
const { contentEl } = this;
|
||||||
|
|
||||||
|
const { authUrl, codeVerifier, codeChallenge } =
|
||||||
|
await generateAuthUrlAndCodeVerifierChallenge(false);
|
||||||
|
this.plugin.oauth2Info.verifier = codeVerifier;
|
||||||
|
|
||||||
|
const t = this.t;
|
||||||
|
|
||||||
|
const div2 = contentEl.createDiv();
|
||||||
|
div2.createEl(
|
||||||
|
"button",
|
||||||
|
{
|
||||||
|
text: t("modal_proauth_copybutton"),
|
||||||
|
},
|
||||||
|
(el) => {
|
||||||
|
el.onclick = async () => {
|
||||||
|
await navigator.clipboard.writeText(authUrl);
|
||||||
|
new Notice(t("modal_proauth_copynotice"));
|
||||||
|
};
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
contentEl.createEl("p").createEl("a", {
|
||||||
|
href: authUrl,
|
||||||
|
text: authUrl,
|
||||||
|
});
|
||||||
|
|
||||||
|
// manual paste
|
||||||
|
let authCode = "";
|
||||||
|
new Setting(contentEl)
|
||||||
|
.setName(t("modal_proauth_maualinput"))
|
||||||
|
.setDesc(t("modal_proauth_maualinput_desc"))
|
||||||
|
.addText((text) =>
|
||||||
|
text
|
||||||
|
.setPlaceholder("")
|
||||||
|
.setValue("")
|
||||||
|
.onChange((val) => {
|
||||||
|
authCode = val.trim();
|
||||||
|
})
|
||||||
|
)
|
||||||
|
.addButton(async (button) => {
|
||||||
|
button.setButtonText(t("submit"));
|
||||||
|
button.onClick(async () => {
|
||||||
|
new Notice(t("modal_proauth_maualinput_notice"));
|
||||||
|
try {
|
||||||
|
const authRes = await sendAuthReq(
|
||||||
|
codeVerifier ?? "verifier",
|
||||||
|
authCode,
|
||||||
|
async (e: any) => {
|
||||||
|
new Notice(t("protocol_pro_connect_fail"));
|
||||||
|
new Notice(`${e}`);
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
console.debug(authRes);
|
||||||
|
const self = this;
|
||||||
|
setConfigBySuccessfullAuthInplace(
|
||||||
|
this.plugin.settings.pro!,
|
||||||
|
authRes!,
|
||||||
|
() => self.plugin.saveSettings()
|
||||||
|
);
|
||||||
|
await getAndSaveProFeatures(
|
||||||
|
this.plugin.settings.pro!,
|
||||||
|
this.plugin.manifest.version,
|
||||||
|
() => self.plugin.saveSettings()
|
||||||
|
);
|
||||||
|
this.proFeaturesListSetting.setDesc(
|
||||||
|
stringToFragment(
|
||||||
|
t("settings_pro_features_desc", {
|
||||||
|
features: featureListToText(
|
||||||
|
this.plugin.settings.pro!.enabledProFeatures
|
||||||
|
),
|
||||||
|
})
|
||||||
|
)
|
||||||
|
);
|
||||||
|
await getAndSaveProEmail(
|
||||||
|
this.plugin.settings.pro!,
|
||||||
|
this.plugin.manifest.version,
|
||||||
|
() => self.plugin.saveSettings()
|
||||||
|
);
|
||||||
|
|
||||||
|
new Notice(
|
||||||
|
t("protocol_pro_connect_manualinput_succ", {
|
||||||
|
email: this.plugin.settings.pro!.email ?? "(no email)",
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
this.plugin.oauth2Info.verifier = ""; // reset it
|
||||||
|
this.plugin.oauth2Info.authDiv?.toggleClass(
|
||||||
|
"pro-auth-button-hide",
|
||||||
|
this.plugin.settings.pro?.refreshToken !== ""
|
||||||
|
);
|
||||||
|
this.plugin.oauth2Info.authDiv = undefined;
|
||||||
|
|
||||||
|
this.plugin.oauth2Info.revokeAuthSetting?.setDesc(
|
||||||
|
t("protocol_pro_connect_succ_revoke", {
|
||||||
|
email: this.plugin.settings.pro?.email,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
this.plugin.oauth2Info.revokeAuthSetting = undefined;
|
||||||
|
this.plugin.oauth2Info.revokeDiv?.toggleClass(
|
||||||
|
"pro-revoke-auth-button-hide",
|
||||||
|
this.plugin.settings.pro?.email === ""
|
||||||
|
);
|
||||||
|
this.plugin.oauth2Info.revokeDiv = undefined;
|
||||||
|
|
||||||
|
this.close();
|
||||||
|
} catch (err) {
|
||||||
|
console.error(err);
|
||||||
|
new Notice(t("modal_proauth_maualinput_conn_fail"));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
onClose() {
|
||||||
|
const { contentEl } = this;
|
||||||
|
contentEl.empty();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class ProRevokeAuthModal extends Modal {
|
||||||
|
readonly plugin: RemotelySavePlugin;
|
||||||
|
readonly authDiv: HTMLDivElement;
|
||||||
|
readonly revokeAuthDiv: HTMLDivElement;
|
||||||
|
readonly t: (x: TransItemType, vars?: any) => string;
|
||||||
|
constructor(
|
||||||
|
app: App,
|
||||||
|
plugin: RemotelySavePlugin,
|
||||||
|
authDiv: HTMLDivElement,
|
||||||
|
revokeAuthDiv: HTMLDivElement,
|
||||||
|
t: (x: TransItemType, vars?: any) => string
|
||||||
|
) {
|
||||||
|
super(app);
|
||||||
|
this.plugin = plugin;
|
||||||
|
this.authDiv = authDiv;
|
||||||
|
this.revokeAuthDiv = revokeAuthDiv;
|
||||||
|
this.t = t;
|
||||||
|
}
|
||||||
|
|
||||||
|
async onOpen() {
|
||||||
|
const { contentEl } = this;
|
||||||
|
const t = this.t;
|
||||||
|
|
||||||
|
contentEl.createEl("p", {
|
||||||
|
text: t("modal_prorevokeauth"),
|
||||||
|
});
|
||||||
|
|
||||||
|
new Setting(contentEl)
|
||||||
|
.setName(t("modal_prorevokeauth_clean"))
|
||||||
|
.setDesc(t("modal_prorevokeauth_clean_desc"))
|
||||||
|
.addButton(async (button) => {
|
||||||
|
button.setButtonText(t("modal_prorevokeauth_clean_button"));
|
||||||
|
button.onClick(async () => {
|
||||||
|
try {
|
||||||
|
this.plugin.settings.pro = cloneDeep(DEFAULT_PRO_CONFIG);
|
||||||
|
await this.plugin.saveSettings();
|
||||||
|
this.authDiv.toggleClass(
|
||||||
|
"pro-auth-button-hide",
|
||||||
|
this.plugin.settings.pro?.refreshToken !== ""
|
||||||
|
);
|
||||||
|
this.revokeAuthDiv.toggleClass(
|
||||||
|
"pro-revoke-auth-button-hide",
|
||||||
|
this.plugin.settings.pro?.refreshToken === ""
|
||||||
|
);
|
||||||
|
new Notice(t("modal_prorevokeauth_clean_notice"));
|
||||||
|
this.close();
|
||||||
|
} catch (err) {
|
||||||
|
console.error(err);
|
||||||
|
new Notice(t("modal_prorevokeauth_clean_fail"));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
onClose() {
|
||||||
|
const { contentEl } = this;
|
||||||
|
contentEl.empty();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const featureListToText = (features: FeatureInfo[]) => {
|
||||||
|
// TODO: i18n
|
||||||
|
if (features === undefined || features.length === 0) {
|
||||||
|
return "No features enabled.";
|
||||||
|
}
|
||||||
|
return features
|
||||||
|
.map((x) => {
|
||||||
|
return `${x.featureName} (expire: ${new Date(
|
||||||
|
Number(x.expireAtTimeMs)
|
||||||
|
).toISOString()})`;
|
||||||
|
})
|
||||||
|
.join("<br/>");
|
||||||
|
};
|
||||||
|
|
||||||
|
export const generateProSettingsPart = (
|
||||||
|
proDiv: HTMLDivElement,
|
||||||
|
t: (x: TransItemType, vars?: any) => string,
|
||||||
|
app: App,
|
||||||
|
plugin: RemotelySavePlugin,
|
||||||
|
saveUpdatedConfigFunc: () => Promise<any> | undefined
|
||||||
|
) => {
|
||||||
|
proDiv
|
||||||
|
.createEl("h2", { text: t("settings_pro") })
|
||||||
|
.setAttribute("id", "settings-pro");
|
||||||
|
|
||||||
|
proDiv.createEl("div", {
|
||||||
|
text: stringToFragment(t("settings_pro_tutorial")),
|
||||||
|
});
|
||||||
|
|
||||||
|
const proSelectAuthDiv = proDiv.createDiv();
|
||||||
|
const proAuthDiv = proSelectAuthDiv.createDiv({
|
||||||
|
cls: "pro-auth-button-hide settings-auth-related",
|
||||||
|
});
|
||||||
|
|
||||||
|
const proRevokeAuthDiv = proSelectAuthDiv.createDiv({
|
||||||
|
cls: "pro-revoke-auth-button-hide settings-auth-related",
|
||||||
|
});
|
||||||
|
|
||||||
|
const proFeaturesListSetting = new Setting(proRevokeAuthDiv)
|
||||||
|
.setName(t("settings_pro_features"))
|
||||||
|
.setDesc(
|
||||||
|
stringToFragment(
|
||||||
|
t("settings_pro_features_desc", {
|
||||||
|
features: featureListToText(plugin.settings.pro!.enabledProFeatures),
|
||||||
|
})
|
||||||
|
)
|
||||||
|
);
|
||||||
|
proFeaturesListSetting.addButton(async (button) => {
|
||||||
|
button.setButtonText(t("settings_pro_features_refresh_button"));
|
||||||
|
button.onClick(async () => {
|
||||||
|
new Notice(t("settings_pro_features_refresh_fetch"));
|
||||||
|
await getAndSaveProFeatures(
|
||||||
|
plugin.settings.pro!,
|
||||||
|
plugin.manifest.version,
|
||||||
|
saveUpdatedConfigFunc
|
||||||
|
);
|
||||||
|
proFeaturesListSetting.setDesc(
|
||||||
|
stringToFragment(
|
||||||
|
t("settings_pro_features_desc", {
|
||||||
|
features: featureListToText(
|
||||||
|
plugin.settings.pro!.enabledProFeatures
|
||||||
|
),
|
||||||
|
})
|
||||||
|
)
|
||||||
|
);
|
||||||
|
new Notice(t("settings_pro_features_refresh_succ"));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
const proRevokeAuthSetting = new Setting(proRevokeAuthDiv)
|
||||||
|
.setName(t("settings_pro_revoke"))
|
||||||
|
.setDesc(
|
||||||
|
t("settings_pro_revoke_desc", {
|
||||||
|
email: plugin.settings.pro?.email,
|
||||||
|
})
|
||||||
|
)
|
||||||
|
.addButton(async (button) => {
|
||||||
|
button.setButtonText(t("settings_pro_revoke_button"));
|
||||||
|
button.onClick(async () => {
|
||||||
|
new ProRevokeAuthModal(
|
||||||
|
app,
|
||||||
|
plugin,
|
||||||
|
proAuthDiv,
|
||||||
|
proRevokeAuthDiv,
|
||||||
|
t
|
||||||
|
).open();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
new Setting(proAuthDiv)
|
||||||
|
.setName(t("settings_pro_intro"))
|
||||||
|
.setDesc(stringToFragment(t("settings_pro_intro_desc")))
|
||||||
|
.addButton(async (button) => {
|
||||||
|
button.setButtonText(t("settings_pro_intro_button"));
|
||||||
|
button.onClick(async () => {
|
||||||
|
window.open("https://remotelysave.com/user/signupin", "_self");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
new Setting(proAuthDiv)
|
||||||
|
.setName(t("settings_pro_auth"))
|
||||||
|
.setDesc(t("settings_pro_auth_desc"))
|
||||||
|
.addButton(async (button) => {
|
||||||
|
button.setButtonText(t("settings_pro_auth_button"));
|
||||||
|
button.onClick(async () => {
|
||||||
|
const modal = new ProAuthModal(
|
||||||
|
app,
|
||||||
|
plugin,
|
||||||
|
proAuthDiv,
|
||||||
|
proRevokeAuthDiv,
|
||||||
|
proRevokeAuthSetting,
|
||||||
|
proFeaturesListSetting,
|
||||||
|
t
|
||||||
|
);
|
||||||
|
plugin.oauth2Info.helperModal = modal;
|
||||||
|
plugin.oauth2Info.authDiv = proAuthDiv;
|
||||||
|
plugin.oauth2Info.revokeDiv = proRevokeAuthDiv;
|
||||||
|
plugin.oauth2Info.revokeAuthSetting = proRevokeAuthSetting;
|
||||||
|
|
||||||
|
modal.open();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
proAuthDiv.toggleClass(
|
||||||
|
"pro-auth-button-hide",
|
||||||
|
plugin.settings.pro?.refreshToken !== ""
|
||||||
|
);
|
||||||
|
proRevokeAuthDiv.toggleClass(
|
||||||
|
"pro-revoke-auth-button-hide",
|
||||||
|
plugin.settings.pro?.refreshToken === ""
|
||||||
|
);
|
||||||
|
};
|
||||||
68
pro/tests/conflictLogic.test.ts
Normal file
68
pro/tests/conflictLogic.test.ts
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
import { deepStrictEqual, rejects, throws } from "assert";
|
||||||
|
import { getFileRename } from "../src/conflictLogic";
|
||||||
|
|
||||||
|
describe("New name is generated", () => {
|
||||||
|
it("should throw for empty file", async () => {
|
||||||
|
for (const key of ["", "/", ".", ".."]) {
|
||||||
|
throws(() => getFileRename(key));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should throw for folder", async () => {
|
||||||
|
for (const key of ["sss/", "ssss/yyy/"]) {
|
||||||
|
throws(() => getFileRename(key));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should correctly get no ext files renamed", async () => {
|
||||||
|
deepStrictEqual(getFileRename("abc"), "abc.dup");
|
||||||
|
|
||||||
|
deepStrictEqual(getFileRename("xxxx/yyyy/abc"), "xxxx/yyyy/abc.dup");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should correctly get dot files renamed", async () => {
|
||||||
|
deepStrictEqual(getFileRename(".abc"), ".abc.dup");
|
||||||
|
|
||||||
|
deepStrictEqual(getFileRename("xxxx/yyyy/.efg"), "xxxx/yyyy/.efg.dup");
|
||||||
|
|
||||||
|
deepStrictEqual(getFileRename("xxxx/yyyy/hij."), "xxxx/yyyy/hij.dup");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should correctly get normal files renamed", async () => {
|
||||||
|
deepStrictEqual(getFileRename("abc.efg"), "abc.dup.efg");
|
||||||
|
|
||||||
|
deepStrictEqual(
|
||||||
|
getFileRename("xxxx/yyyy/abc.efg"),
|
||||||
|
"xxxx/yyyy/abc.dup.efg"
|
||||||
|
);
|
||||||
|
|
||||||
|
deepStrictEqual(
|
||||||
|
getFileRename("xxxx/yyyy/abc.tar.gz"),
|
||||||
|
"xxxx/yyyy/abc.tar.dup.gz"
|
||||||
|
);
|
||||||
|
|
||||||
|
deepStrictEqual(
|
||||||
|
getFileRename("xxxx/yyyy/.abc.efg"),
|
||||||
|
"xxxx/yyyy/.abc.dup.efg"
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should correctly get duplicated files renamed again", async () => {
|
||||||
|
deepStrictEqual(getFileRename("abc.dup"), "abc.dup.dup");
|
||||||
|
|
||||||
|
deepStrictEqual(
|
||||||
|
getFileRename("xxxx/yyyy/.abc.dup"),
|
||||||
|
"xxxx/yyyy/.abc.dup.dup"
|
||||||
|
);
|
||||||
|
|
||||||
|
deepStrictEqual(
|
||||||
|
getFileRename("xxxx/yyyy/abc.dup.md"),
|
||||||
|
"xxxx/yyyy/abc.dup.dup.md"
|
||||||
|
);
|
||||||
|
|
||||||
|
deepStrictEqual(
|
||||||
|
getFileRename("xxxx/yyyy/.abc.dup.md"),
|
||||||
|
"xxxx/yyyy/.abc.dup.dup.md"
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
201
src/LICENSE
Normal file
201
src/LICENSE
Normal file
@ -0,0 +1,201 @@
|
|||||||
|
Apache License
|
||||||
|
Version 2.0, January 2004
|
||||||
|
http://www.apache.org/licenses/
|
||||||
|
|
||||||
|
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
||||||
|
|
||||||
|
1. Definitions.
|
||||||
|
|
||||||
|
"License" shall mean the terms and conditions for use, reproduction,
|
||||||
|
and distribution as defined by Sections 1 through 9 of this document.
|
||||||
|
|
||||||
|
"Licensor" shall mean the copyright owner or entity authorized by
|
||||||
|
the copyright owner that is granting the License.
|
||||||
|
|
||||||
|
"Legal Entity" shall mean the union of the acting entity and all
|
||||||
|
other entities that control, are controlled by, or are under common
|
||||||
|
control with that entity. For the purposes of this definition,
|
||||||
|
"control" means (i) the power, direct or indirect, to cause the
|
||||||
|
direction or management of such entity, whether by contract or
|
||||||
|
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
||||||
|
outstanding shares, or (iii) beneficial ownership of such entity.
|
||||||
|
|
||||||
|
"You" (or "Your") shall mean an individual or Legal Entity
|
||||||
|
exercising permissions granted by this License.
|
||||||
|
|
||||||
|
"Source" form shall mean the preferred form for making modifications,
|
||||||
|
including but not limited to software source code, documentation
|
||||||
|
source, and configuration files.
|
||||||
|
|
||||||
|
"Object" form shall mean any form resulting from mechanical
|
||||||
|
transformation or translation of a Source form, including but
|
||||||
|
not limited to compiled object code, generated documentation,
|
||||||
|
and conversions to other media types.
|
||||||
|
|
||||||
|
"Work" shall mean the work of authorship, whether in Source or
|
||||||
|
Object form, made available under the License, as indicated by a
|
||||||
|
copyright notice that is included in or attached to the work
|
||||||
|
(an example is provided in the Appendix below).
|
||||||
|
|
||||||
|
"Derivative Works" shall mean any work, whether in Source or Object
|
||||||
|
form, that is based on (or derived from) the Work and for which the
|
||||||
|
editorial revisions, annotations, elaborations, or other modifications
|
||||||
|
represent, as a whole, an original work of authorship. For the purposes
|
||||||
|
of this License, Derivative Works shall not include works that remain
|
||||||
|
separable from, or merely link (or bind by name) to the interfaces of,
|
||||||
|
the Work and Derivative Works thereof.
|
||||||
|
|
||||||
|
"Contribution" shall mean any work of authorship, including
|
||||||
|
the original version of the Work and any modifications or additions
|
||||||
|
to that Work or Derivative Works thereof, that is intentionally
|
||||||
|
submitted to Licensor for inclusion in the Work by the copyright owner
|
||||||
|
or by an individual or Legal Entity authorized to submit on behalf of
|
||||||
|
the copyright owner. For the purposes of this definition, "submitted"
|
||||||
|
means any form of electronic, verbal, or written communication sent
|
||||||
|
to the Licensor or its representatives, including but not limited to
|
||||||
|
communication on electronic mailing lists, source code control systems,
|
||||||
|
and issue tracking systems that are managed by, or on behalf of, the
|
||||||
|
Licensor for the purpose of discussing and improving the Work, but
|
||||||
|
excluding communication that is conspicuously marked or otherwise
|
||||||
|
designated in writing by the copyright owner as "Not a Contribution."
|
||||||
|
|
||||||
|
"Contributor" shall mean Licensor and any individual or Legal Entity
|
||||||
|
on behalf of whom a Contribution has been received by Licensor and
|
||||||
|
subsequently incorporated within the Work.
|
||||||
|
|
||||||
|
2. Grant of Copyright License. Subject to the terms and conditions of
|
||||||
|
this License, each Contributor hereby grants to You a perpetual,
|
||||||
|
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||||
|
copyright license to reproduce, prepare Derivative Works of,
|
||||||
|
publicly display, publicly perform, sublicense, and distribute the
|
||||||
|
Work and such Derivative Works in Source or Object form.
|
||||||
|
|
||||||
|
3. Grant of Patent License. Subject to the terms and conditions of
|
||||||
|
this License, each Contributor hereby grants to You a perpetual,
|
||||||
|
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||||
|
(except as stated in this section) patent license to make, have made,
|
||||||
|
use, offer to sell, sell, import, and otherwise transfer the Work,
|
||||||
|
where such license applies only to those patent claims licensable
|
||||||
|
by such Contributor that are necessarily infringed by their
|
||||||
|
Contribution(s) alone or by combination of their Contribution(s)
|
||||||
|
with the Work to which such Contribution(s) was submitted. If You
|
||||||
|
institute patent litigation against any entity (including a
|
||||||
|
cross-claim or counterclaim in a lawsuit) alleging that the Work
|
||||||
|
or a Contribution incorporated within the Work constitutes direct
|
||||||
|
or contributory patent infringement, then any patent licenses
|
||||||
|
granted to You under this License for that Work shall terminate
|
||||||
|
as of the date such litigation is filed.
|
||||||
|
|
||||||
|
4. Redistribution. You may reproduce and distribute copies of the
|
||||||
|
Work or Derivative Works thereof in any medium, with or without
|
||||||
|
modifications, and in Source or Object form, provided that You
|
||||||
|
meet the following conditions:
|
||||||
|
|
||||||
|
(a) You must give any other recipients of the Work or
|
||||||
|
Derivative Works a copy of this License; and
|
||||||
|
|
||||||
|
(b) You must cause any modified files to carry prominent notices
|
||||||
|
stating that You changed the files; and
|
||||||
|
|
||||||
|
(c) You must retain, in the Source form of any Derivative Works
|
||||||
|
that You distribute, all copyright, patent, trademark, and
|
||||||
|
attribution notices from the Source form of the Work,
|
||||||
|
excluding those notices that do not pertain to any part of
|
||||||
|
the Derivative Works; and
|
||||||
|
|
||||||
|
(d) If the Work includes a "NOTICE" text file as part of its
|
||||||
|
distribution, then any Derivative Works that You distribute must
|
||||||
|
include a readable copy of the attribution notices contained
|
||||||
|
within such NOTICE file, excluding those notices that do not
|
||||||
|
pertain to any part of the Derivative Works, in at least one
|
||||||
|
of the following places: within a NOTICE text file distributed
|
||||||
|
as part of the Derivative Works; within the Source form or
|
||||||
|
documentation, if provided along with the Derivative Works; or,
|
||||||
|
within a display generated by the Derivative Works, if and
|
||||||
|
wherever such third-party notices normally appear. The contents
|
||||||
|
of the NOTICE file are for informational purposes only and
|
||||||
|
do not modify the License. You may add Your own attribution
|
||||||
|
notices within Derivative Works that You distribute, alongside
|
||||||
|
or as an addendum to the NOTICE text from the Work, provided
|
||||||
|
that such additional attribution notices cannot be construed
|
||||||
|
as modifying the License.
|
||||||
|
|
||||||
|
You may add Your own copyright statement to Your modifications and
|
||||||
|
may provide additional or different license terms and conditions
|
||||||
|
for use, reproduction, or distribution of Your modifications, or
|
||||||
|
for any such Derivative Works as a whole, provided Your use,
|
||||||
|
reproduction, and distribution of the Work otherwise complies with
|
||||||
|
the conditions stated in this License.
|
||||||
|
|
||||||
|
5. Submission of Contributions. Unless You explicitly state otherwise,
|
||||||
|
any Contribution intentionally submitted for inclusion in the Work
|
||||||
|
by You to the Licensor shall be under the terms and conditions of
|
||||||
|
this License, without any additional terms or conditions.
|
||||||
|
Notwithstanding the above, nothing herein shall supersede or modify
|
||||||
|
the terms of any separate license agreement you may have executed
|
||||||
|
with Licensor regarding such Contributions.
|
||||||
|
|
||||||
|
6. Trademarks. This License does not grant permission to use the trade
|
||||||
|
names, trademarks, service marks, or product names of the Licensor,
|
||||||
|
except as required for reasonable and customary use in describing the
|
||||||
|
origin of the Work and reproducing the content of the NOTICE file.
|
||||||
|
|
||||||
|
7. Disclaimer of Warranty. Unless required by applicable law or
|
||||||
|
agreed to in writing, Licensor provides the Work (and each
|
||||||
|
Contributor provides its Contributions) on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
||||||
|
implied, including, without limitation, any warranties or conditions
|
||||||
|
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
|
||||||
|
PARTICULAR PURPOSE. You are solely responsible for determining the
|
||||||
|
appropriateness of using or redistributing the Work and assume any
|
||||||
|
risks associated with Your exercise of permissions under this License.
|
||||||
|
|
||||||
|
8. Limitation of Liability. In no event and under no legal theory,
|
||||||
|
whether in tort (including negligence), contract, or otherwise,
|
||||||
|
unless required by applicable law (such as deliberate and grossly
|
||||||
|
negligent acts) or agreed to in writing, shall any Contributor be
|
||||||
|
liable to You for damages, including any direct, indirect, special,
|
||||||
|
incidental, or consequential damages of any character arising as a
|
||||||
|
result of this License or out of the use or inability to use the
|
||||||
|
Work (including but not limited to damages for loss of goodwill,
|
||||||
|
work stoppage, computer failure or malfunction, or any and all
|
||||||
|
other commercial damages or losses), even if such Contributor
|
||||||
|
has been advised of the possibility of such damages.
|
||||||
|
|
||||||
|
9. Accepting Warranty or Additional Liability. While redistributing
|
||||||
|
the Work or Derivative Works thereof, You may choose to offer,
|
||||||
|
and charge a fee for, acceptance of support, warranty, indemnity,
|
||||||
|
or other liability obligations and/or rights consistent with this
|
||||||
|
License. However, in accepting such obligations, You may act only
|
||||||
|
on Your own behalf and on Your sole responsibility, not on behalf
|
||||||
|
of any other Contributor, and only if You agree to indemnify,
|
||||||
|
defend, and hold each Contributor harmless for any liability
|
||||||
|
incurred by, or claims asserted against, such Contributor by reason
|
||||||
|
of your accepting any such warranty or additional liability.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
APPENDIX: How to apply the Apache License to your work.
|
||||||
|
|
||||||
|
To apply the Apache License to your work, attach the following
|
||||||
|
boilerplate notice, with the fields enclosed by brackets "[]"
|
||||||
|
replaced with your own identifying information. (Don't include
|
||||||
|
the brackets!) The text should be enclosed in the appropriate
|
||||||
|
comment syntax for the file format. We also recommend that a
|
||||||
|
file or class name and description of purpose be included on the
|
||||||
|
same "printed page" as the copyright notice for easier
|
||||||
|
identification within third-party archives.
|
||||||
|
|
||||||
|
Copyright [yyyy] [name of copyright owner]
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
9
src/README.md
Normal file
9
src/README.md
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
# Main Basic Source
|
||||||
|
|
||||||
|
## What?
|
||||||
|
|
||||||
|
The main basic source code for Remotely Save.
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
The codes or files or subfolders inside the current folder (`src` in the repo), are released under "open source" license: "Apache License, version 2.0".
|
||||||
@ -3,6 +3,7 @@
|
|||||||
* To avoid circular dependency.
|
* To avoid circular dependency.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import type { ProConfig } from "../pro/src/baseTypesPro";
|
||||||
import type { LangTypeAndAuto } from "./i18n";
|
import type { LangTypeAndAuto } from "./i18n";
|
||||||
|
|
||||||
export const DEFAULT_CONTENT_TYPE = "application/octet-stream";
|
export const DEFAULT_CONTENT_TYPE = "application/octet-stream";
|
||||||
@ -155,6 +156,8 @@ export interface RemotelySavePluginSettings {
|
|||||||
|
|
||||||
profiler?: ProfilerConfig;
|
profiler?: ProfilerConfig;
|
||||||
|
|
||||||
|
pro?: ProConfig;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @deprecated
|
* @deprecated
|
||||||
*/
|
*/
|
||||||
@ -188,7 +191,10 @@ export const OAUTH2_FORCE_EXPIRE_MILLISECONDS = 1000 * 60 * 60 * 24 * 80;
|
|||||||
|
|
||||||
export type EmptyFolderCleanType = "skip" | "clean_both";
|
export type EmptyFolderCleanType = "skip" | "clean_both";
|
||||||
|
|
||||||
export type ConflictActionType = "keep_newer" | "keep_larger" | "rename_both";
|
export type ConflictActionType =
|
||||||
|
| "keep_newer"
|
||||||
|
| "keep_larger"
|
||||||
|
| "smart_conflict";
|
||||||
|
|
||||||
export type DecisionTypeForMixedEntity =
|
export type DecisionTypeForMixedEntity =
|
||||||
| "only_history"
|
| "only_history"
|
||||||
@ -203,11 +209,11 @@ export type DecisionTypeForMixedEntity =
|
|||||||
| "remote_is_deleted_thus_also_delete_local"
|
| "remote_is_deleted_thus_also_delete_local"
|
||||||
| "conflict_created_then_keep_local"
|
| "conflict_created_then_keep_local"
|
||||||
| "conflict_created_then_keep_remote"
|
| "conflict_created_then_keep_remote"
|
||||||
| "conflict_created_then_keep_both"
|
| "conflict_created_then_smart_conflict"
|
||||||
| "conflict_created_then_do_nothing"
|
| "conflict_created_then_do_nothing"
|
||||||
| "conflict_modified_then_keep_local"
|
| "conflict_modified_then_keep_local"
|
||||||
| "conflict_modified_then_keep_remote"
|
| "conflict_modified_then_keep_remote"
|
||||||
| "conflict_modified_then_keep_both"
|
| "conflict_modified_then_smart_conflict"
|
||||||
| "folder_existed_both_then_do_nothing"
|
| "folder_existed_both_then_do_nothing"
|
||||||
| "folder_existed_local_then_also_create_remote"
|
| "folder_existed_local_then_also_create_remote"
|
||||||
| "folder_existed_remote_then_also_create_local"
|
| "folder_existed_remote_then_also_create_local"
|
||||||
|
|||||||
60
src/copyLogic.ts
Normal file
60
src/copyLogic.ts
Normal file
@ -0,0 +1,60 @@
|
|||||||
|
import type { FakeFs } from "./fsAll";
|
||||||
|
|
||||||
|
export async function copyFolder(key: string, left: FakeFs, right: FakeFs) {
|
||||||
|
if (!key.endsWith("/")) {
|
||||||
|
throw Error(`should not call ${key} in copyFolder`);
|
||||||
|
}
|
||||||
|
const statsLeft = await left.stat(key);
|
||||||
|
const entity = await right.mkdir(key, statsLeft.mtimeCli);
|
||||||
|
return {
|
||||||
|
entity: entity,
|
||||||
|
content: undefined,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function copyFile(key: string, left: FakeFs, right: FakeFs) {
|
||||||
|
// console.debug(`copyFile: key=${key}, left=${left.kind}, right=${right.kind}`);
|
||||||
|
if (key.endsWith("/")) {
|
||||||
|
throw Error(`should not call ${key} in copyFile`);
|
||||||
|
}
|
||||||
|
const statsLeft = await left.stat(key);
|
||||||
|
const content = await left.readFile(key);
|
||||||
|
|
||||||
|
if (statsLeft.size === undefined || statsLeft.size === 0) {
|
||||||
|
// some weird bugs on android not returning size. just ignore them
|
||||||
|
statsLeft.size = content.byteLength;
|
||||||
|
} else {
|
||||||
|
if (statsLeft.size !== content.byteLength) {
|
||||||
|
throw Error(
|
||||||
|
`error copying ${left.kind}=>${right.kind}: size not matched`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (statsLeft.mtimeCli === undefined) {
|
||||||
|
throw Error(`error copying ${left.kind}=>${right.kind}, no mtimeCli`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// console.debug(`copyFile: about to start right.writeFile`);
|
||||||
|
return {
|
||||||
|
entity: await right.writeFile(
|
||||||
|
key,
|
||||||
|
content,
|
||||||
|
statsLeft.mtimeCli,
|
||||||
|
statsLeft.mtimeCli /* TODO */
|
||||||
|
),
|
||||||
|
content: content,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function copyFileOrFolder(
|
||||||
|
key: string,
|
||||||
|
left: FakeFs,
|
||||||
|
right: FakeFs
|
||||||
|
) {
|
||||||
|
if (key.endsWith("/")) {
|
||||||
|
return await copyFolder(key, left, right);
|
||||||
|
} else {
|
||||||
|
return await copyFile(key, left, right);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -13,6 +13,7 @@ export abstract class FakeFs {
|
|||||||
ctime: number
|
ctime: number
|
||||||
): Promise<Entity>;
|
): Promise<Entity>;
|
||||||
abstract readFile(key: string): Promise<ArrayBuffer>;
|
abstract readFile(key: string): Promise<ArrayBuffer>;
|
||||||
|
abstract rename(key1: string, key2: string): Promise<void>;
|
||||||
abstract rm(key: string): Promise<void>;
|
abstract rm(key: string): Promise<void>;
|
||||||
abstract checkConnect(callbackFunc?: any): Promise<boolean>;
|
abstract checkConnect(callbackFunc?: any): Promise<boolean>;
|
||||||
abstract getUserDisplayName(): Promise<string>;
|
abstract getUserDisplayName(): Promise<string>;
|
||||||
|
|||||||
@ -695,6 +695,25 @@ export class FakeFsDropbox extends FakeFs {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async rename(key1: string, key2: string): Promise<void> {
|
||||||
|
const remoteFileName1 = getDropboxPath(key1, this.remoteBaseDir);
|
||||||
|
const remoteFileName2 = getDropboxPath(key2, this.remoteBaseDir);
|
||||||
|
await this._init();
|
||||||
|
try {
|
||||||
|
await retryReq(
|
||||||
|
() =>
|
||||||
|
this.dropbox.filesMoveV2({
|
||||||
|
from_path: remoteFileName1,
|
||||||
|
to_path: remoteFileName2,
|
||||||
|
}),
|
||||||
|
`${key1}=>${key2}` // just a hint here
|
||||||
|
);
|
||||||
|
} catch (err) {
|
||||||
|
console.error("some error while moving");
|
||||||
|
console.error(err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
async rm(key: string): Promise<void> {
|
async rm(key: string): Promise<void> {
|
||||||
if (key === "/") {
|
if (key === "/") {
|
||||||
return;
|
return;
|
||||||
|
|||||||
@ -356,6 +356,31 @@ export class FakeFsEncrypt extends FakeFs {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async rename(key1: string, key2: string): Promise<void> {
|
||||||
|
if (!this.hasCacheMap) {
|
||||||
|
throw new Error("You have to build the cacheMap firstly for readFile");
|
||||||
|
}
|
||||||
|
let key1Enc = this.cacheMapOrigToEnc[key1];
|
||||||
|
if (key1Enc === undefined) {
|
||||||
|
if (this.isPasswordEmpty()) {
|
||||||
|
key1Enc = key1;
|
||||||
|
} else {
|
||||||
|
key1Enc = await this._encryptName(key1);
|
||||||
|
}
|
||||||
|
this.cacheMapOrigToEnc[key1] = key1Enc;
|
||||||
|
}
|
||||||
|
let key2Enc = this.cacheMapOrigToEnc[key2];
|
||||||
|
if (key2Enc === undefined) {
|
||||||
|
if (this.isPasswordEmpty()) {
|
||||||
|
key2Enc = key2;
|
||||||
|
} else {
|
||||||
|
key2Enc = await this._encryptName(key2);
|
||||||
|
}
|
||||||
|
this.cacheMapOrigToEnc[key2] = key2Enc;
|
||||||
|
}
|
||||||
|
return await this.innerFs.rename(key1Enc, key2Enc);
|
||||||
|
}
|
||||||
|
|
||||||
async rm(key: string): Promise<void> {
|
async rm(key: string): Promise<void> {
|
||||||
if (!this.hasCacheMap) {
|
if (!this.hasCacheMap) {
|
||||||
throw new Error("You have to build the cacheMap firstly for rm");
|
throw new Error("You have to build the cacheMap firstly for rm");
|
||||||
|
|||||||
@ -145,6 +145,7 @@ export class FakeFsLocal extends FakeFs {
|
|||||||
): Promise<Entity> {
|
): Promise<Entity> {
|
||||||
await this.vault.adapter.writeBinary(key, content, {
|
await this.vault.adapter.writeBinary(key, content, {
|
||||||
mtime: mtime,
|
mtime: mtime,
|
||||||
|
ctime: ctime,
|
||||||
});
|
});
|
||||||
return await this.stat(key);
|
return await this.stat(key);
|
||||||
}
|
}
|
||||||
@ -153,6 +154,10 @@ export class FakeFsLocal extends FakeFs {
|
|||||||
return await this.vault.adapter.readBinary(key);
|
return await this.vault.adapter.readBinary(key);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async rename(key1: string, key2: string): Promise<void> {
|
||||||
|
return await this.vault.adapter.rename(key1, key2);
|
||||||
|
}
|
||||||
|
|
||||||
async rm(key: string): Promise<void> {
|
async rm(key: string): Promise<void> {
|
||||||
if (this.deleteToWhere === "obsidian") {
|
if (this.deleteToWhere === "obsidian") {
|
||||||
await this.vault.adapter.trashLocal(key);
|
await this.vault.adapter.trashLocal(key);
|
||||||
|
|||||||
@ -38,6 +38,10 @@ export class FakeFsMock extends FakeFs {
|
|||||||
throw new Error("Method not implemented.");
|
throw new Error("Method not implemented.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async rename(key1: string, key2: string): Promise<void> {
|
||||||
|
throw new Error("Method not implemented.");
|
||||||
|
}
|
||||||
|
|
||||||
async rm(key: string): Promise<void> {
|
async rm(key: string): Promise<void> {
|
||||||
throw new Error("Method not implemented.");
|
throw new Error("Method not implemented.");
|
||||||
}
|
}
|
||||||
|
|||||||
@ -928,6 +928,18 @@ export class FakeFsOnedrive extends FakeFs {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async rename(key1: string, key2: string): Promise<void> {
|
||||||
|
if (key1 === "" || key1 === "/" || key2 === "" || key2 === "/") {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const remoteFileName1 = getOnedrivePath(key1, this.remoteBaseDir);
|
||||||
|
const remoteFileName2 = getOnedrivePath(key2, this.remoteBaseDir);
|
||||||
|
await this._init();
|
||||||
|
await this._patchJson(remoteFileName1, {
|
||||||
|
name: remoteFileName2,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
async rm(key: string): Promise<void> {
|
async rm(key: string): Promise<void> {
|
||||||
if (key === "" || key === "/") {
|
if (key === "" || key === "/") {
|
||||||
return;
|
return;
|
||||||
|
|||||||
@ -746,6 +746,10 @@ export class FakeFsS3 extends FakeFs {
|
|||||||
return bodyContents;
|
return bodyContents;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async rename(key1: string, key2: string): Promise<void> {
|
||||||
|
throw Error(`rename not implemented for s3`);
|
||||||
|
}
|
||||||
|
|
||||||
async rm(key: string): Promise<void> {
|
async rm(key: string): Promise<void> {
|
||||||
if (key === "/") {
|
if (key === "/") {
|
||||||
return;
|
return;
|
||||||
|
|||||||
@ -66,24 +66,11 @@ if (VALID_REQURL) {
|
|||||||
const p: RequestUrlParam = {
|
const p: RequestUrlParam = {
|
||||||
url: options.url,
|
url: options.url,
|
||||||
method: options.method,
|
method: options.method,
|
||||||
// body: options.data as string | ArrayBuffer,
|
body: options.data as string | ArrayBuffer,
|
||||||
headers: transformedHeaders,
|
headers: transformedHeaders,
|
||||||
contentType: reqContentType,
|
contentType: reqContentType,
|
||||||
throw: false,
|
throw: false,
|
||||||
};
|
};
|
||||||
if (
|
|
||||||
options.data === undefined ||
|
|
||||||
options.data === null ||
|
|
||||||
isString(options.data)
|
|
||||||
) {
|
|
||||||
p.body = options.data;
|
|
||||||
} else {
|
|
||||||
if (typeof (options.data as any).transfer === "function") {
|
|
||||||
p.body = (options.data as any).transfer();
|
|
||||||
} else {
|
|
||||||
p.body = options.data as ArrayBuffer;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let r = await requestUrl(p);
|
let r = await requestUrl(p);
|
||||||
|
|
||||||
@ -330,9 +317,9 @@ export class FakeFsWebdav extends FakeFs {
|
|||||||
*/
|
*/
|
||||||
_getnextcloudUploadServerAddress = () => {
|
_getnextcloudUploadServerAddress = () => {
|
||||||
let k = this.webdavConfig.address;
|
let k = this.webdavConfig.address;
|
||||||
if (k.endsWith('/')) {
|
if (k.endsWith("/")) {
|
||||||
// no tailing slash
|
// no tailing slash
|
||||||
k = k.substring(0, k.length-1);
|
k = k.substring(0, k.length - 1);
|
||||||
}
|
}
|
||||||
const s = k.split("/");
|
const s = k.split("/");
|
||||||
if (
|
if (
|
||||||
@ -797,6 +784,16 @@ export class FakeFsWebdav extends FakeFs {
|
|||||||
throw Error(`unexpected file content result with type ${typeof buff}`);
|
throw Error(`unexpected file content result with type ${typeof buff}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async rename(key1: string, key2: string): Promise<void> {
|
||||||
|
if (key1 === "/" || key2 === "/") {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const remoteFileName1 = getWebdavPath(key1, this.remoteBaseDir);
|
||||||
|
const remoteFileName2 = getWebdavPath(key2, this.remoteBaseDir);
|
||||||
|
await this._init();
|
||||||
|
await this.client.moveFile(remoteFileName1, remoteFileName2);
|
||||||
|
}
|
||||||
|
|
||||||
async rm(key: string): Promise<void> {
|
async rm(key: string): Promise<void> {
|
||||||
if (key === "/") {
|
if (key === "/") {
|
||||||
return;
|
return;
|
||||||
|
|||||||
@ -230,6 +230,15 @@ export class FakeFsWebdis extends FakeFs {
|
|||||||
return rsp;
|
return rsp;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async rename(key1: string, key2: string): Promise<void> {
|
||||||
|
const fullKey1 = getWebdisPath(key1, this.remoteBaseDir);
|
||||||
|
const fullKey2 = getWebdisPath(key2, this.remoteBaseDir);
|
||||||
|
const commandContent = `RENAME/${fullKey1}:content/${fullKey2}:content`;
|
||||||
|
await this._fetchCommand("POST", commandContent);
|
||||||
|
const commandMeta = `RENAME/${fullKey1}:meta/${fullKey2}:meta`;
|
||||||
|
await this._fetchCommand("POST", commandMeta);
|
||||||
|
}
|
||||||
|
|
||||||
async rm(key: string): Promise<void> {
|
async rm(key: string): Promise<void> {
|
||||||
const fullKey = getWebdisPath(key, this.remoteBaseDir);
|
const fullKey = getWebdisPath(key, this.remoteBaseDir);
|
||||||
const command = `DEL/${fullKey}:meta/${fullKey}:content`;
|
const command = `DEL/${fullKey}:meta/${fullKey}:content`;
|
||||||
|
|||||||
@ -1,7 +1,11 @@
|
|||||||
|
import merge from "lodash/merge";
|
||||||
import Mustache from "mustache";
|
import Mustache from "mustache";
|
||||||
import { moment } from "obsidian";
|
import { moment } from "obsidian";
|
||||||
|
|
||||||
import { LANGS } from "./langs";
|
import { LANGS as LANGS_PRO } from "../pro/src/langs";
|
||||||
|
import { LANGS as LANGS_BASIC } from "./langs";
|
||||||
|
|
||||||
|
const LANGS = merge(LANGS_BASIC, LANGS_PRO);
|
||||||
|
|
||||||
export type LangType = keyof typeof LANGS;
|
export type LangType = keyof typeof LANGS;
|
||||||
export type LangTypeAndAuto = LangType | "auto";
|
export type LangTypeAndAuto = LangType | "auto";
|
||||||
|
|||||||
@ -24,6 +24,7 @@ export const exportQrCodeUri = async (
|
|||||||
delete settings2.onedrive;
|
delete settings2.onedrive;
|
||||||
delete settings2.webdav;
|
delete settings2.webdav;
|
||||||
delete settings2.webdis;
|
delete settings2.webdis;
|
||||||
|
delete settings2.pro;
|
||||||
} else if (exportFields === "s3") {
|
} else if (exportFields === "s3") {
|
||||||
settings2 = { s3: cloneDeep(settings.s3) };
|
settings2 = { s3: cloneDeep(settings.s3) };
|
||||||
} else if (exportFields === "dropbox") {
|
} else if (exportFields === "dropbox") {
|
||||||
|
|||||||
@ -277,7 +277,7 @@
|
|||||||
"settings_deletetowhere_system_trash": "system trash (default)",
|
"settings_deletetowhere_system_trash": "system trash (default)",
|
||||||
"settings_deletetowhere_obsidian_trash": "Obsidian .trash folder",
|
"settings_deletetowhere_obsidian_trash": "Obsidian .trash folder",
|
||||||
"settings_conflictaction": "Action For Conflict",
|
"settings_conflictaction": "Action For Conflict",
|
||||||
"settings_conflictaction_desc": "If a file is created or modified on both side since last update, it's a conflict event. How to deal with it? This only works for bidirectional sync.",
|
"settings_conflictaction_desc": "<p>If a file is created or modified on both side since last update, it's a conflict event. How to deal with it? This only works for bidirectional sync.</p>",
|
||||||
"settings_conflictaction_keep_newer": "newer version survives (default)",
|
"settings_conflictaction_keep_newer": "newer version survives (default)",
|
||||||
"settings_conflictaction_keep_larger": "larger size version survives",
|
"settings_conflictaction_keep_larger": "larger size version survives",
|
||||||
"settings_cleanemptyfolder": "Action For Empty Folders",
|
"settings_cleanemptyfolder": "Action For Empty Folders",
|
||||||
|
|||||||
@ -20,6 +20,7 @@ export const DEFAULT_TBL_LOGGER_OUTPUT = "loggeroutput";
|
|||||||
export const DEFAULT_TBL_SIMPLE_KV_FOR_MISC = "simplekvformisc";
|
export const DEFAULT_TBL_SIMPLE_KV_FOR_MISC = "simplekvformisc";
|
||||||
export const DEFAULT_TBL_PREV_SYNC_RECORDS = "prevsyncrecords";
|
export const DEFAULT_TBL_PREV_SYNC_RECORDS = "prevsyncrecords";
|
||||||
export const DEFAULT_TBL_PROFILER_RESULTS = "profilerresults";
|
export const DEFAULT_TBL_PROFILER_RESULTS = "profilerresults";
|
||||||
|
export const DEFAULT_TBL_FILE_CONTENT_HISTORY = "filecontenthistory";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @deprecated
|
* @deprecated
|
||||||
@ -62,6 +63,7 @@ export interface InternalDBs {
|
|||||||
simpleKVForMiscTbl: LocalForage;
|
simpleKVForMiscTbl: LocalForage;
|
||||||
prevSyncRecordsTbl: LocalForage;
|
prevSyncRecordsTbl: LocalForage;
|
||||||
profilerResultsTbl: LocalForage;
|
profilerResultsTbl: LocalForage;
|
||||||
|
fileContentHistoryTbl: LocalForage;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @deprecated
|
* @deprecated
|
||||||
@ -221,6 +223,11 @@ export const prepareDBs = async (
|
|||||||
name: DEFAULT_DB_NAME,
|
name: DEFAULT_DB_NAME,
|
||||||
storeName: DEFAULT_TBL_SYNC_MAPPING,
|
storeName: DEFAULT_TBL_SYNC_MAPPING,
|
||||||
}),
|
}),
|
||||||
|
|
||||||
|
fileContentHistoryTbl: localforage.createInstance({
|
||||||
|
name: DEFAULT_DB_NAME,
|
||||||
|
storeName: DEFAULT_TBL_FILE_CONTENT_HISTORY,
|
||||||
|
}),
|
||||||
} as InternalDBs;
|
} as InternalDBs;
|
||||||
|
|
||||||
// try to get vaultRandomID firstly
|
// try to get vaultRandomID firstly
|
||||||
|
|||||||
88
src/main.ts
88
src/main.ts
@ -13,6 +13,13 @@ import {
|
|||||||
requireApiVersion,
|
requireApiVersion,
|
||||||
setIcon,
|
setIcon,
|
||||||
} from "obsidian";
|
} from "obsidian";
|
||||||
|
import {
|
||||||
|
DEFAULT_PRO_CONFIG,
|
||||||
|
getAndSaveProEmail,
|
||||||
|
getAndSaveProFeatures,
|
||||||
|
sendAuthReq as sendAuthReqPro,
|
||||||
|
setConfigBySuccessfullAuthInplace as setConfigBySuccessfullAuthInplacePro,
|
||||||
|
} from "../pro/src/account";
|
||||||
import type {
|
import type {
|
||||||
RemotelySavePluginSettings,
|
RemotelySavePluginSettings,
|
||||||
SyncTriggerSourceType,
|
SyncTriggerSourceType,
|
||||||
@ -56,6 +63,7 @@ import { SyncAlgoV3Modal } from "./syncAlgoV3Notice";
|
|||||||
// biome-ignore lint/suspicious/noShadowRestrictedNames: <explanation>
|
// biome-ignore lint/suspicious/noShadowRestrictedNames: <explanation>
|
||||||
import AggregateError from "aggregate-error";
|
import AggregateError from "aggregate-error";
|
||||||
import throttle from "lodash/throttle";
|
import throttle from "lodash/throttle";
|
||||||
|
import { COMMAND_CALLBACK_PRO } from "../pro/src/baseTypesPro";
|
||||||
import { exportVaultSyncPlansToFiles } from "./debugMode";
|
import { exportVaultSyncPlansToFiles } from "./debugMode";
|
||||||
import { FakeFsEncrypt } from "./fsEncrypt";
|
import { FakeFsEncrypt } from "./fsEncrypt";
|
||||||
import { getClient } from "./fsGetter";
|
import { getClient } from "./fsGetter";
|
||||||
@ -97,6 +105,7 @@ const DEFAULT_SETTINGS: RemotelySavePluginSettings = {
|
|||||||
enableMobileStatusBar: false,
|
enableMobileStatusBar: false,
|
||||||
encryptionMethod: "unknown",
|
encryptionMethod: "unknown",
|
||||||
profiler: DEFAULT_PROFILER_CONFIG,
|
profiler: DEFAULT_PROFILER_CONFIG,
|
||||||
|
pro: DEFAULT_PRO_CONFIG,
|
||||||
};
|
};
|
||||||
|
|
||||||
interface OAuth2Info {
|
interface OAuth2Info {
|
||||||
@ -399,6 +408,8 @@ export default class RemotelySavePlugin extends Plugin {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const configSaver = async () => await this.saveSettings();
|
||||||
|
|
||||||
await syncer(
|
await syncer(
|
||||||
fsLocal,
|
fsLocal,
|
||||||
fsRemote,
|
fsRemote,
|
||||||
@ -410,6 +421,8 @@ export default class RemotelySavePlugin extends Plugin {
|
|||||||
this.vaultRandomID,
|
this.vaultRandomID,
|
||||||
this.app.vault.configDir,
|
this.app.vault.configDir,
|
||||||
this.settings,
|
this.settings,
|
||||||
|
this.manifest.version,
|
||||||
|
configSaver,
|
||||||
getProtectError,
|
getProtectError,
|
||||||
markIsSyncingFunc,
|
markIsSyncingFunc,
|
||||||
notifyFunc,
|
notifyFunc,
|
||||||
@ -696,6 +709,77 @@ export default class RemotelySavePlugin extends Plugin {
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
this.registerObsidianProtocolHandler(
|
||||||
|
COMMAND_CALLBACK_PRO,
|
||||||
|
async (inputParams) => {
|
||||||
|
if (this.oauth2Info.helperModal !== undefined) {
|
||||||
|
const k = this.oauth2Info.helperModal.contentEl;
|
||||||
|
k.empty();
|
||||||
|
|
||||||
|
t("protocol_pro_connecting")
|
||||||
|
.split("\n")
|
||||||
|
.forEach((val) => {
|
||||||
|
k.createEl("p", {
|
||||||
|
text: val,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
console.debug(inputParams);
|
||||||
|
const authRes = await sendAuthReqPro(
|
||||||
|
this.oauth2Info.verifier || "verifier",
|
||||||
|
inputParams.code,
|
||||||
|
async (e: any) => {
|
||||||
|
new Notice(t("protocol_pro_connect_fail"));
|
||||||
|
new Notice(`${e}`);
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
console.debug(authRes);
|
||||||
|
|
||||||
|
const self = this;
|
||||||
|
await setConfigBySuccessfullAuthInplacePro(
|
||||||
|
this.settings.pro!,
|
||||||
|
authRes,
|
||||||
|
() => self.saveSettings()
|
||||||
|
);
|
||||||
|
|
||||||
|
await getAndSaveProFeatures(
|
||||||
|
this.settings.pro!,
|
||||||
|
this.manifest.version,
|
||||||
|
() => self.saveSettings()
|
||||||
|
);
|
||||||
|
|
||||||
|
await getAndSaveProEmail(
|
||||||
|
this.settings.pro!,
|
||||||
|
this.manifest.version,
|
||||||
|
() => self.saveSettings()
|
||||||
|
);
|
||||||
|
|
||||||
|
this.oauth2Info.verifier = ""; // reset it
|
||||||
|
this.oauth2Info.helperModal?.close(); // close it
|
||||||
|
this.oauth2Info.helperModal = undefined;
|
||||||
|
|
||||||
|
this.oauth2Info.authDiv?.toggleClass(
|
||||||
|
"pro-auth-button-hide",
|
||||||
|
this.settings.pro?.refreshToken !== ""
|
||||||
|
);
|
||||||
|
this.oauth2Info.authDiv = undefined;
|
||||||
|
|
||||||
|
this.oauth2Info.revokeAuthSetting?.setDesc(
|
||||||
|
t("protocol_pro_connect_succ_revoke", {
|
||||||
|
email: this.settings.pro?.email,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
this.oauth2Info.revokeAuthSetting = undefined;
|
||||||
|
this.oauth2Info.revokeDiv?.toggleClass(
|
||||||
|
"pro-revoke-auth-button-hide",
|
||||||
|
this.settings.pro?.email === ""
|
||||||
|
);
|
||||||
|
this.oauth2Info.revokeDiv = undefined;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
this.syncRibbon = this.addRibbonIcon(
|
this.syncRibbon = this.addRibbonIcon(
|
||||||
iconNameSyncWait,
|
iconNameSyncWait,
|
||||||
`${this.manifest.name}`,
|
`${this.manifest.name}`,
|
||||||
@ -1046,6 +1130,10 @@ export default class RemotelySavePlugin extends Plugin {
|
|||||||
needSave = true;
|
needSave = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (this.settings.pro === undefined) {
|
||||||
|
this.settings.pro = cloneDeep(DEFAULT_PRO_CONFIG);
|
||||||
|
}
|
||||||
|
|
||||||
// save back
|
// save back
|
||||||
if (needSave) {
|
if (needSave) {
|
||||||
await this.saveSettings();
|
await this.saveSettings();
|
||||||
|
|||||||
@ -416,7 +416,7 @@ export const toText = (x: any) => {
|
|||||||
export const statFix = async (vault: Vault, path: string) => {
|
export const statFix = async (vault: Vault, path: string) => {
|
||||||
const s = await vault.adapter.stat(path);
|
const s = await vault.adapter.stat(path);
|
||||||
if (s === undefined || s === null) {
|
if (s === undefined || s === null) {
|
||||||
return s;
|
throw Error(`${path} doesn't exist cannot run stat`);
|
||||||
}
|
}
|
||||||
if (s.ctime === undefined || s.ctime === null || Number.isNaN(s.ctime)) {
|
if (s.ctime === undefined || s.ctime === null || Number.isNaN(s.ctime)) {
|
||||||
s.ctime = undefined as any; // force assignment
|
s.ctime = undefined as any; // force assignment
|
||||||
|
|||||||
@ -21,6 +21,7 @@ import type {
|
|||||||
} from "./baseTypes";
|
} from "./baseTypes";
|
||||||
|
|
||||||
import cloneDeep from "lodash/cloneDeep";
|
import cloneDeep from "lodash/cloneDeep";
|
||||||
|
import { generateProSettingsPart } from "../pro/src/settingsPro";
|
||||||
import { API_VER_ENSURE_REQURL_OK, VALID_REQURL } from "./baseTypesObs";
|
import { API_VER_ENSURE_REQURL_OK, VALID_REQURL } from "./baseTypesObs";
|
||||||
import { messyConfigToNormal } from "./configPersist";
|
import { messyConfigToNormal } from "./configPersist";
|
||||||
import {
|
import {
|
||||||
@ -2130,25 +2131,44 @@ export class RemotelySaveSettingTab extends PluginSettingTab {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
new Setting(advDiv)
|
let conflictActionSettingOrigDesc = t("settings_conflictaction_desc");
|
||||||
|
if (
|
||||||
|
(this.plugin.settings.conflictAction ?? "keep_newer") === "smart_conflict"
|
||||||
|
) {
|
||||||
|
conflictActionSettingOrigDesc += t(
|
||||||
|
"settings_conflictaction_smart_conflict_desc"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
const conflictActionSetting = new Setting(advDiv)
|
||||||
.setName(t("settings_conflictaction"))
|
.setName(t("settings_conflictaction"))
|
||||||
.setDesc(t("settings_conflictaction_desc"))
|
.setDesc(stringToFragment(conflictActionSettingOrigDesc));
|
||||||
.addDropdown((dropdown) => {
|
conflictActionSetting.addDropdown((dropdown) => {
|
||||||
dropdown.addOption(
|
dropdown
|
||||||
"keep_newer",
|
.addOption("keep_newer", t("settings_conflictaction_keep_newer"))
|
||||||
t("settings_conflictaction_keep_newer")
|
.addOption("keep_larger", t("settings_conflictaction_keep_larger"))
|
||||||
);
|
.addOption(
|
||||||
dropdown.addOption(
|
"smart_conflict",
|
||||||
"keep_larger",
|
t("settings_conflictaction_smart_conflict")
|
||||||
t("settings_conflictaction_keep_larger")
|
)
|
||||||
);
|
.setValue(this.plugin.settings.conflictAction ?? "keep_newer")
|
||||||
dropdown
|
.onChange(async (val) => {
|
||||||
.setValue(this.plugin.settings.conflictAction ?? "keep_newer")
|
this.plugin.settings.conflictAction = val as ConflictActionType;
|
||||||
.onChange(async (val) => {
|
await this.plugin.saveSettings();
|
||||||
this.plugin.settings.conflictAction = val as ConflictActionType;
|
|
||||||
await this.plugin.saveSettings();
|
conflictActionSettingOrigDesc = t("settings_conflictaction_desc");
|
||||||
});
|
if (
|
||||||
});
|
(this.plugin.settings.conflictAction ?? "keep_newer") ===
|
||||||
|
"smart_conflict"
|
||||||
|
) {
|
||||||
|
conflictActionSettingOrigDesc += t(
|
||||||
|
"settings_conflictaction_smart_conflict_desc"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
conflictActionSetting.setDesc(
|
||||||
|
stringToFragment(conflictActionSettingOrigDesc)
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
new Setting(advDiv)
|
new Setting(advDiv)
|
||||||
.setName(t("settings_cleanemptyfolder"))
|
.setName(t("settings_cleanemptyfolder"))
|
||||||
@ -2417,6 +2437,15 @@ export class RemotelySaveSettingTab extends PluginSettingTab {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////
|
||||||
|
// below for pro
|
||||||
|
//////////////////////////////////////////////////
|
||||||
|
|
||||||
|
const proDiv = containerEl.createEl("div");
|
||||||
|
generateProSettingsPart(proDiv, t, this.app, this.plugin, () =>
|
||||||
|
this.plugin.saveSettings()
|
||||||
|
);
|
||||||
|
|
||||||
//////////////////////////////////////////////////
|
//////////////////////////////////////////////////
|
||||||
// below for debug
|
// below for debug
|
||||||
//////////////////////////////////////////////////
|
//////////////////////////////////////////////////
|
||||||
|
|||||||
310
src/sync.ts
310
src/sync.ts
@ -2,6 +2,13 @@
|
|||||||
import AggregateError from "aggregate-error";
|
import AggregateError from "aggregate-error";
|
||||||
import PQueue from "p-queue";
|
import PQueue from "p-queue";
|
||||||
import XRegExp from "xregexp";
|
import XRegExp from "xregexp";
|
||||||
|
import { checkProRunnableAndFixInplace } from "../pro/src/account";
|
||||||
|
import { duplicateFile, isMergable, mergeFile } from "../pro/src/conflictLogic";
|
||||||
|
import {
|
||||||
|
clearFileContentHistoryByVaultAndProfile,
|
||||||
|
getFileContentHistoryByVaultAndProfile,
|
||||||
|
upsertFileContentHistoryByVaultAndProfile,
|
||||||
|
} from "../pro/src/localdb";
|
||||||
import type {
|
import type {
|
||||||
ConflictActionType,
|
ConflictActionType,
|
||||||
EmptyFolderCleanType,
|
EmptyFolderCleanType,
|
||||||
@ -12,6 +19,7 @@ import type {
|
|||||||
SyncDirectionType,
|
SyncDirectionType,
|
||||||
SyncTriggerSourceType,
|
SyncTriggerSourceType,
|
||||||
} from "./baseTypes";
|
} from "./baseTypes";
|
||||||
|
import { copyFile, copyFileOrFolder, copyFolder } from "./copyLogic";
|
||||||
import type { FakeFs } from "./fsAll";
|
import type { FakeFs } from "./fsAll";
|
||||||
import type { FakeFsEncrypt } from "./fsEncrypt";
|
import type { FakeFsEncrypt } from "./fsEncrypt";
|
||||||
import {
|
import {
|
||||||
@ -432,7 +440,6 @@ const getSyncPlanInplace = async (
|
|||||||
} else {
|
} else {
|
||||||
// Both exists, but modified or conflict
|
// Both exists, but modified or conflict
|
||||||
// Look for past files of A or B.
|
// Look for past files of A or B.
|
||||||
|
|
||||||
const localEqualPrevSync =
|
const localEqualPrevSync =
|
||||||
prevSync?.mtimeCli === local.mtimeCli &&
|
prevSync?.mtimeCli === local.mtimeCli &&
|
||||||
prevSync?.sizeEnc === local.sizeEnc;
|
prevSync?.sizeEnc === local.sizeEnc;
|
||||||
@ -521,9 +528,10 @@ const getSyncPlanInplace = async (
|
|||||||
mixedEntry.change = true;
|
mixedEntry.change = true;
|
||||||
keptFolder.add(getParentFolder(key));
|
keptFolder.add(getParentFolder(key));
|
||||||
}
|
}
|
||||||
} else {
|
} else if (conflictAction === "smart_conflict") {
|
||||||
mixedEntry.decisionBranch = 15;
|
// try merge!
|
||||||
mixedEntry.decision = "conflict_created_then_keep_both";
|
mixedEntry.decisionBranch = 302;
|
||||||
|
mixedEntry.decision = "conflict_created_then_smart_conflict";
|
||||||
mixedEntry.change = true;
|
mixedEntry.change = true;
|
||||||
keptFolder.add(getParentFolder(key));
|
keptFolder.add(getParentFolder(key));
|
||||||
}
|
}
|
||||||
@ -572,9 +580,10 @@ const getSyncPlanInplace = async (
|
|||||||
mixedEntry.change = true;
|
mixedEntry.change = true;
|
||||||
keptFolder.add(getParentFolder(key));
|
keptFolder.add(getParentFolder(key));
|
||||||
}
|
}
|
||||||
} else {
|
} else if (conflictAction === "smart_conflict") {
|
||||||
mixedEntry.decisionBranch = 20;
|
// yeah, try to merge them!
|
||||||
mixedEntry.decision = "conflict_modified_then_keep_both";
|
mixedEntry.decisionBranch = 301;
|
||||||
|
mixedEntry.decision = "conflict_modified_then_smart_conflict";
|
||||||
mixedEntry.change = true;
|
mixedEntry.change = true;
|
||||||
keptFolder.add(getParentFolder(key));
|
keptFolder.add(getParentFolder(key));
|
||||||
}
|
}
|
||||||
@ -900,10 +909,10 @@ const splitFourStepsOnEntityMappings = (
|
|||||||
val.decision === "remote_is_created_then_pull" ||
|
val.decision === "remote_is_created_then_pull" ||
|
||||||
val.decision === "conflict_created_then_keep_local" ||
|
val.decision === "conflict_created_then_keep_local" ||
|
||||||
val.decision === "conflict_created_then_keep_remote" ||
|
val.decision === "conflict_created_then_keep_remote" ||
|
||||||
val.decision === "conflict_created_then_keep_both" ||
|
val.decision === "conflict_created_then_smart_conflict" ||
|
||||||
val.decision === "conflict_modified_then_keep_local" ||
|
val.decision === "conflict_modified_then_keep_local" ||
|
||||||
val.decision === "conflict_modified_then_keep_remote" ||
|
val.decision === "conflict_modified_then_keep_remote" ||
|
||||||
val.decision === "conflict_modified_then_keep_both"
|
val.decision === "conflict_modified_then_smart_conflict"
|
||||||
) {
|
) {
|
||||||
if (
|
if (
|
||||||
uploadDownloads.length === 0 ||
|
uploadDownloads.length === 0 ||
|
||||||
@ -966,75 +975,6 @@ const fullfillMTimeOfRemoteEntityInplace = (
|
|||||||
return remote;
|
return remote;
|
||||||
};
|
};
|
||||||
|
|
||||||
async function copyFolder(
|
|
||||||
key: string,
|
|
||||||
left: FakeFs,
|
|
||||||
right: FakeFs
|
|
||||||
): Promise<Entity> {
|
|
||||||
if (!key.endsWith("/")) {
|
|
||||||
throw Error(`should not call ${key} in copyFolder`);
|
|
||||||
}
|
|
||||||
const statsLeft = await left.stat(key);
|
|
||||||
return await right.mkdir(key, statsLeft.mtimeCli);
|
|
||||||
}
|
|
||||||
|
|
||||||
async function copyFile(
|
|
||||||
key: string,
|
|
||||||
left: FakeFs,
|
|
||||||
right: FakeFs
|
|
||||||
): Promise<Entity> {
|
|
||||||
// console.debug(`copyFile: key=${key}, left=${left.kind}, right=${right.kind}`);
|
|
||||||
if (key.endsWith("/")) {
|
|
||||||
throw Error(`should not call ${key} in copyFile`);
|
|
||||||
}
|
|
||||||
const statsLeft = await left.stat(key);
|
|
||||||
const content = await left.readFile(key);
|
|
||||||
|
|
||||||
if (statsLeft.size === undefined || statsLeft.size === 0) {
|
|
||||||
// some weird bugs on android not returning size. just ignore them
|
|
||||||
statsLeft.size = content.byteLength;
|
|
||||||
} else {
|
|
||||||
if (statsLeft.size !== content.byteLength) {
|
|
||||||
throw Error(
|
|
||||||
`error copying ${left.kind}=>${right.kind}: size not matched`
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (statsLeft.mtimeCli === undefined) {
|
|
||||||
throw Error(`error copying ${left.kind}=>${right.kind}, no mtimeCli`);
|
|
||||||
}
|
|
||||||
|
|
||||||
// console.debug(`copyFile: about to start right.writeFile`);
|
|
||||||
if (typeof (content as any).transfer === "function") {
|
|
||||||
return await right.writeFile(
|
|
||||||
key,
|
|
||||||
(content as any).transfer(),
|
|
||||||
statsLeft.mtimeCli,
|
|
||||||
statsLeft.mtimeCli /* TODO */
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
return await right.writeFile(
|
|
||||||
key,
|
|
||||||
content,
|
|
||||||
statsLeft.mtimeCli,
|
|
||||||
statsLeft.mtimeCli /* TODO */
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async function copyFileOrFolder(
|
|
||||||
key: string,
|
|
||||||
left: FakeFs,
|
|
||||||
right: FakeFs
|
|
||||||
): Promise<Entity> {
|
|
||||||
if (key.endsWith("/")) {
|
|
||||||
return await copyFolder(key, left, right);
|
|
||||||
} else {
|
|
||||||
return await copyFile(key, left, right);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const dispatchOperationToActualV3 = async (
|
const dispatchOperationToActualV3 = async (
|
||||||
key: string,
|
key: string,
|
||||||
vaultRandomID: string,
|
vaultRandomID: string,
|
||||||
@ -1042,7 +982,8 @@ const dispatchOperationToActualV3 = async (
|
|||||||
r: MixedEntity,
|
r: MixedEntity,
|
||||||
fsLocal: FakeFs,
|
fsLocal: FakeFs,
|
||||||
fsEncrypt: FakeFsEncrypt,
|
fsEncrypt: FakeFsEncrypt,
|
||||||
db: InternalDBs
|
db: InternalDBs,
|
||||||
|
conflictAction: ConflictActionType
|
||||||
) => {
|
) => {
|
||||||
// console.debug(
|
// console.debug(
|
||||||
// `inside dispatchOperationToActualV3, key=${key}, r=${JSON.stringify(
|
// `inside dispatchOperationToActualV3, key=${key}, r=${JSON.stringify(
|
||||||
@ -1052,7 +993,20 @@ const dispatchOperationToActualV3 = async (
|
|||||||
// )}`
|
// )}`
|
||||||
// );
|
// );
|
||||||
if (r.decision === "only_history") {
|
if (r.decision === "only_history") {
|
||||||
clearPrevSyncRecordByVaultAndProfile(db, vaultRandomID, profileID, key);
|
await clearPrevSyncRecordByVaultAndProfile(
|
||||||
|
db,
|
||||||
|
vaultRandomID,
|
||||||
|
profileID,
|
||||||
|
key
|
||||||
|
);
|
||||||
|
if (conflictAction === "smart_conflict") {
|
||||||
|
await clearFileContentHistoryByVaultAndProfile(
|
||||||
|
db,
|
||||||
|
vaultRandomID,
|
||||||
|
profileID,
|
||||||
|
key
|
||||||
|
);
|
||||||
|
}
|
||||||
} else if (
|
} else if (
|
||||||
r.decision === "local_is_created_too_large_then_do_nothing" ||
|
r.decision === "local_is_created_too_large_then_do_nothing" ||
|
||||||
r.decision === "remote_is_created_too_large_then_do_nothing" ||
|
r.decision === "remote_is_created_too_large_then_do_nothing" ||
|
||||||
@ -1071,12 +1025,34 @@ const dispatchOperationToActualV3 = async (
|
|||||||
|
|
||||||
if (r.prevSync !== undefined) {
|
if (r.prevSync !== undefined) {
|
||||||
// if we have prevSync,
|
// if we have prevSync,
|
||||||
// we don't need to do anything, because the record is already there!
|
// we don't need to update prevSync, because the record is already there!
|
||||||
|
|
||||||
|
// but we might need to update content, because it's a new feature
|
||||||
|
if (conflictAction === "smart_conflict") {
|
||||||
|
if (isMergable(r.local!)) {
|
||||||
|
const k = await getFileContentHistoryByVaultAndProfile(
|
||||||
|
db,
|
||||||
|
vaultRandomID,
|
||||||
|
profileID,
|
||||||
|
r.local!
|
||||||
|
);
|
||||||
|
if (k === null || k === undefined) {
|
||||||
|
await upsertFileContentHistoryByVaultAndProfile(
|
||||||
|
db,
|
||||||
|
vaultRandomID,
|
||||||
|
profileID,
|
||||||
|
r.local!,
|
||||||
|
await fsLocal.readFile(r.local!.keyRaw)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
// if we don't have prevSync, we use remote entity AND local mtime
|
// if we don't have prevSync, we use remote entity AND local mtime
|
||||||
// as if it is "uploaded"
|
// as if it is "uploaded"
|
||||||
if (r.remote !== undefined) {
|
if (r.remote !== undefined) {
|
||||||
let entity = r.remote;
|
let entity = r.remote;
|
||||||
|
// TODO: abstract away the dirty hack
|
||||||
entity = fullfillMTimeOfRemoteEntityInplace(entity, r.local?.mtimeCli);
|
entity = fullfillMTimeOfRemoteEntityInplace(entity, r.local?.mtimeCli);
|
||||||
|
|
||||||
if (entity !== undefined) {
|
if (entity !== undefined) {
|
||||||
@ -1086,6 +1062,17 @@ const dispatchOperationToActualV3 = async (
|
|||||||
profileID,
|
profileID,
|
||||||
entity
|
entity
|
||||||
);
|
);
|
||||||
|
if (conflictAction === "smart_conflict") {
|
||||||
|
if (isMergable(entity)) {
|
||||||
|
await upsertFileContentHistoryByVaultAndProfile(
|
||||||
|
db,
|
||||||
|
vaultRandomID,
|
||||||
|
profileID,
|
||||||
|
entity,
|
||||||
|
await fsLocal.readFile(entity.keyRaw)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1098,7 +1085,12 @@ const dispatchOperationToActualV3 = async (
|
|||||||
) {
|
) {
|
||||||
// console.debug(`before upload in sync, r=${JSON.stringify(r, null, 2)}`);
|
// console.debug(`before upload in sync, r=${JSON.stringify(r, null, 2)}`);
|
||||||
const mtimeCli = (await fsLocal.stat(r.key)).mtimeCli!;
|
const mtimeCli = (await fsLocal.stat(r.key)).mtimeCli!;
|
||||||
const entity = await copyFileOrFolder(r.key, fsLocal, fsEncrypt);
|
const { entity, content } = await copyFileOrFolder(
|
||||||
|
r.key,
|
||||||
|
fsLocal,
|
||||||
|
fsEncrypt
|
||||||
|
);
|
||||||
|
// TODO: abstract away the dirty hack
|
||||||
fullfillMTimeOfRemoteEntityInplace(entity, mtimeCli);
|
fullfillMTimeOfRemoteEntityInplace(entity, mtimeCli);
|
||||||
// console.debug(`after fullfill, entity=${JSON.stringify(entity,null,2)}`)
|
// console.debug(`after fullfill, entity=${JSON.stringify(entity,null,2)}`)
|
||||||
await upsertPrevSyncRecordByVaultAndProfile(
|
await upsertPrevSyncRecordByVaultAndProfile(
|
||||||
@ -1107,6 +1099,17 @@ const dispatchOperationToActualV3 = async (
|
|||||||
profileID,
|
profileID,
|
||||||
entity
|
entity
|
||||||
);
|
);
|
||||||
|
if (conflictAction === "smart_conflict") {
|
||||||
|
if (isMergable(entity)) {
|
||||||
|
await upsertFileContentHistoryByVaultAndProfile(
|
||||||
|
db,
|
||||||
|
vaultRandomID,
|
||||||
|
profileID,
|
||||||
|
entity,
|
||||||
|
content!
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
} else if (
|
} else if (
|
||||||
r.decision === "remote_is_modified_then_pull" ||
|
r.decision === "remote_is_modified_then_pull" ||
|
||||||
r.decision === "remote_is_created_then_pull" ||
|
r.decision === "remote_is_created_then_pull" ||
|
||||||
@ -1114,10 +1117,14 @@ const dispatchOperationToActualV3 = async (
|
|||||||
r.decision === "conflict_modified_then_keep_remote" ||
|
r.decision === "conflict_modified_then_keep_remote" ||
|
||||||
r.decision === "folder_existed_remote_then_also_create_local"
|
r.decision === "folder_existed_remote_then_also_create_local"
|
||||||
) {
|
) {
|
||||||
|
let e1: Entity | undefined = undefined;
|
||||||
|
let c1: ArrayBuffer | undefined = undefined;
|
||||||
if (r.key.endsWith("/")) {
|
if (r.key.endsWith("/")) {
|
||||||
await fsLocal.mkdir(r.key);
|
await fsLocal.mkdir(r.key);
|
||||||
} else {
|
} else {
|
||||||
await copyFile(r.key, fsEncrypt, fsLocal);
|
const { entity, content } = await copyFile(r.key, fsEncrypt, fsLocal);
|
||||||
|
e1 = entity;
|
||||||
|
c1 = content;
|
||||||
}
|
}
|
||||||
await upsertPrevSyncRecordByVaultAndProfile(
|
await upsertPrevSyncRecordByVaultAndProfile(
|
||||||
db,
|
db,
|
||||||
@ -1125,6 +1132,17 @@ const dispatchOperationToActualV3 = async (
|
|||||||
profileID,
|
profileID,
|
||||||
r.remote!
|
r.remote!
|
||||||
);
|
);
|
||||||
|
if (conflictAction === "smart_conflict") {
|
||||||
|
if (isMergable(r.remote!)) {
|
||||||
|
await upsertFileContentHistoryByVaultAndProfile(
|
||||||
|
db,
|
||||||
|
vaultRandomID,
|
||||||
|
profileID,
|
||||||
|
r.remote!,
|
||||||
|
c1! // always file, always has real value
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
} else if (r.decision === "local_is_deleted_thus_also_delete_remote") {
|
} else if (r.decision === "local_is_deleted_thus_also_delete_remote") {
|
||||||
// local is deleted, we need to delete remote now
|
// local is deleted, we need to delete remote now
|
||||||
await fsEncrypt.rm(r.key);
|
await fsEncrypt.rm(r.key);
|
||||||
@ -1134,6 +1152,16 @@ const dispatchOperationToActualV3 = async (
|
|||||||
profileID,
|
profileID,
|
||||||
r.key
|
r.key
|
||||||
);
|
);
|
||||||
|
if (conflictAction === "smart_conflict") {
|
||||||
|
if (isMergable(r.remote!)) {
|
||||||
|
await clearFileContentHistoryByVaultAndProfile(
|
||||||
|
db,
|
||||||
|
vaultRandomID,
|
||||||
|
profileID,
|
||||||
|
r.key
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
} else if (r.decision === "remote_is_deleted_thus_also_delete_local") {
|
} else if (r.decision === "remote_is_deleted_thus_also_delete_local") {
|
||||||
// remote is deleted, we need to delete local now
|
// remote is deleted, we need to delete local now
|
||||||
await fsLocal.rm(r.key);
|
await fsLocal.rm(r.key);
|
||||||
@ -1143,20 +1171,92 @@ const dispatchOperationToActualV3 = async (
|
|||||||
profileID,
|
profileID,
|
||||||
r.key
|
r.key
|
||||||
);
|
);
|
||||||
|
if (conflictAction === "smart_conflict") {
|
||||||
|
if (isMergable(r.remote!)) {
|
||||||
|
await clearFileContentHistoryByVaultAndProfile(
|
||||||
|
db,
|
||||||
|
vaultRandomID,
|
||||||
|
profileID,
|
||||||
|
r.key
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
} else if (
|
} else if (
|
||||||
r.decision === "conflict_created_then_keep_both" ||
|
r.decision === "conflict_created_then_smart_conflict" ||
|
||||||
r.decision === "conflict_modified_then_keep_both"
|
r.decision === "conflict_modified_then_smart_conflict"
|
||||||
) {
|
) {
|
||||||
throw Error(`${r.decision} not implemented yet: ${JSON.stringify(r)}`);
|
// heavy lifting
|
||||||
|
if (isMergable(r.local!, r.remote!)) {
|
||||||
|
const origContent = await getFileContentHistoryByVaultAndProfile(
|
||||||
|
db,
|
||||||
|
vaultRandomID,
|
||||||
|
profileID,
|
||||||
|
r.local!
|
||||||
|
);
|
||||||
|
// console.debug(`we get origContent:`)
|
||||||
|
// console.debug(origContent)
|
||||||
|
const { entity, content } = await mergeFile(
|
||||||
|
r.key,
|
||||||
|
fsLocal,
|
||||||
|
fsEncrypt,
|
||||||
|
origContent
|
||||||
|
);
|
||||||
|
await upsertPrevSyncRecordByVaultAndProfile(
|
||||||
|
db,
|
||||||
|
vaultRandomID,
|
||||||
|
profileID,
|
||||||
|
entity
|
||||||
|
);
|
||||||
|
await upsertFileContentHistoryByVaultAndProfile(
|
||||||
|
db,
|
||||||
|
vaultRandomID,
|
||||||
|
profileID,
|
||||||
|
entity,
|
||||||
|
content
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
// duplicate the files
|
||||||
|
await clearPrevSyncRecordByVaultAndProfile(
|
||||||
|
db,
|
||||||
|
vaultRandomID,
|
||||||
|
profileID,
|
||||||
|
r.key
|
||||||
|
);
|
||||||
|
const mtimeCli = (await fsLocal.stat(r.key)).mtimeCli!;
|
||||||
|
const { upload, download } = await duplicateFile(
|
||||||
|
r.key,
|
||||||
|
fsLocal,
|
||||||
|
fsEncrypt,
|
||||||
|
async (upload) => {
|
||||||
|
// TODO: abstract away the dirty hack
|
||||||
|
fullfillMTimeOfRemoteEntityInplace(upload, mtimeCli);
|
||||||
|
await upsertPrevSyncRecordByVaultAndProfile(
|
||||||
|
db,
|
||||||
|
vaultRandomID,
|
||||||
|
profileID,
|
||||||
|
upload
|
||||||
|
);
|
||||||
|
},
|
||||||
|
async (download) => {
|
||||||
|
await upsertPrevSyncRecordByVaultAndProfile(
|
||||||
|
db,
|
||||||
|
vaultRandomID,
|
||||||
|
profileID,
|
||||||
|
download
|
||||||
|
);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
} else if (r.decision === "folder_to_be_created") {
|
} else if (r.decision === "folder_to_be_created") {
|
||||||
await fsLocal.mkdir(r.key);
|
await fsLocal.mkdir(r.key);
|
||||||
const entity = await copyFolder(r.key, fsLocal, fsEncrypt);
|
const { entity } = await copyFolder(r.key, fsLocal, fsEncrypt);
|
||||||
await upsertPrevSyncRecordByVaultAndProfile(
|
await upsertPrevSyncRecordByVaultAndProfile(
|
||||||
db,
|
db,
|
||||||
vaultRandomID,
|
vaultRandomID,
|
||||||
profileID,
|
profileID,
|
||||||
entity
|
entity
|
||||||
);
|
);
|
||||||
|
// no need to record file content for folder here
|
||||||
} else if (
|
} else if (
|
||||||
r.decision === "folder_to_be_deleted_on_both" ||
|
r.decision === "folder_to_be_deleted_on_both" ||
|
||||||
r.decision === "folder_to_be_deleted_on_local" ||
|
r.decision === "folder_to_be_deleted_on_local" ||
|
||||||
@ -1180,6 +1280,7 @@ const dispatchOperationToActualV3 = async (
|
|||||||
profileID,
|
profileID,
|
||||||
r.key
|
r.key
|
||||||
);
|
);
|
||||||
|
// no need to record file content for folder here
|
||||||
} else {
|
} else {
|
||||||
throw Error(`don't know how to dispatch decision: ${JSON.stringify(r)}`);
|
throw Error(`don't know how to dispatch decision: ${JSON.stringify(r)}`);
|
||||||
}
|
}
|
||||||
@ -1196,6 +1297,7 @@ export const doActualSync = async (
|
|||||||
getProtectModifyPercentageErrorStrFunc: any,
|
getProtectModifyPercentageErrorStrFunc: any,
|
||||||
db: InternalDBs,
|
db: InternalDBs,
|
||||||
profiler: Profiler | undefined,
|
profiler: Profiler | undefined,
|
||||||
|
conflictAction: ConflictActionType,
|
||||||
callbackSyncProcess?: any
|
callbackSyncProcess?: any
|
||||||
) => {
|
) => {
|
||||||
profiler?.addIndent();
|
profiler?.addIndent();
|
||||||
@ -1318,7 +1420,8 @@ export const doActualSync = async (
|
|||||||
val,
|
val,
|
||||||
fsLocal,
|
fsLocal,
|
||||||
fsEncrypt,
|
fsEncrypt,
|
||||||
db
|
db,
|
||||||
|
conflictAction
|
||||||
);
|
);
|
||||||
|
|
||||||
// console.debug(`finished ${key}`);
|
// console.debug(`finished ${key}`);
|
||||||
@ -1381,6 +1484,8 @@ export async function syncer(
|
|||||||
vaultRandomID: string,
|
vaultRandomID: string,
|
||||||
configDir: string,
|
configDir: string,
|
||||||
settings: RemotelySavePluginSettings,
|
settings: RemotelySavePluginSettings,
|
||||||
|
pluginVersion: string,
|
||||||
|
configSaver: () => Promise<any>,
|
||||||
getProtectModifyPercentageErrorStrFunc: any,
|
getProtectModifyPercentageErrorStrFunc: any,
|
||||||
markIsSyncingFunc: (isSyncing: boolean) => void,
|
markIsSyncingFunc: (isSyncing: boolean) => void,
|
||||||
notifyFunc?: (s: SyncTriggerSourceType, step: number) => Promise<any>,
|
notifyFunc?: (s: SyncTriggerSourceType, step: number) => Promise<any>,
|
||||||
@ -1397,17 +1502,27 @@ export async function syncer(
|
|||||||
markIsSyncingFunc(true);
|
markIsSyncingFunc(true);
|
||||||
|
|
||||||
let everythingOk = true;
|
let everythingOk = true;
|
||||||
|
let step = 0;
|
||||||
let step = 0; // dry mode only
|
|
||||||
await notifyFunc?.(triggerSource, step);
|
|
||||||
|
|
||||||
step = 1;
|
|
||||||
await notifyFunc?.(triggerSource, step);
|
|
||||||
await ribboonFunc?.(triggerSource, step);
|
|
||||||
await statusBarFunc?.(triggerSource, step, everythingOk);
|
|
||||||
profiler?.insert("start big sync func");
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
// check pro feature
|
||||||
|
// if anything goes wrong, it will throw
|
||||||
|
await checkProRunnableAndFixInplace(
|
||||||
|
["feature-smart_conflict"],
|
||||||
|
settings,
|
||||||
|
pluginVersion,
|
||||||
|
configSaver
|
||||||
|
);
|
||||||
|
|
||||||
|
// try mode?
|
||||||
|
await notifyFunc?.(triggerSource, step);
|
||||||
|
|
||||||
|
step = 1;
|
||||||
|
await notifyFunc?.(triggerSource, step);
|
||||||
|
await ribboonFunc?.(triggerSource, step);
|
||||||
|
await statusBarFunc?.(triggerSource, step, everythingOk);
|
||||||
|
profiler?.insert("start big sync func");
|
||||||
|
|
||||||
step = 2;
|
step = 2;
|
||||||
await notifyFunc?.(triggerSource, step);
|
await notifyFunc?.(triggerSource, step);
|
||||||
await ribboonFunc?.(triggerSource, step);
|
await ribboonFunc?.(triggerSource, step);
|
||||||
@ -1514,6 +1629,7 @@ export async function syncer(
|
|||||||
getProtectModifyPercentageErrorStrFunc,
|
getProtectModifyPercentageErrorStrFunc,
|
||||||
db,
|
db,
|
||||||
profiler,
|
profiler,
|
||||||
|
settings.conflictAction ?? "keep_newer",
|
||||||
callbackSyncProcess
|
callbackSyncProcess
|
||||||
);
|
);
|
||||||
profiler?.insert(`finish step${step} (actual sync)`);
|
profiler?.insert(`finish step${step} (actual sync)`);
|
||||||
|
|||||||
15
styles.css
15
styles.css
@ -90,3 +90,18 @@
|
|||||||
/* flex-wrap: wrap; */
|
/* flex-wrap: wrap; */
|
||||||
display: grid;
|
display: grid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.pro-disclaimer {
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
.pro-hide {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pro-auth-button-hide {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pro-revoke-auth-button-hide {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|||||||
@ -6,6 +6,8 @@ const TerserPlugin = require("terser-webpack-plugin");
|
|||||||
const DEFAULT_DROPBOX_APP_KEY = process.env.DROPBOX_APP_KEY || "";
|
const DEFAULT_DROPBOX_APP_KEY = process.env.DROPBOX_APP_KEY || "";
|
||||||
const DEFAULT_ONEDRIVE_CLIENT_ID = process.env.ONEDRIVE_CLIENT_ID || "";
|
const DEFAULT_ONEDRIVE_CLIENT_ID = process.env.ONEDRIVE_CLIENT_ID || "";
|
||||||
const DEFAULT_ONEDRIVE_AUTHORITY = process.env.ONEDRIVE_AUTHORITY || "";
|
const DEFAULT_ONEDRIVE_AUTHORITY = process.env.ONEDRIVE_AUTHORITY || "";
|
||||||
|
const DEFAULT_REMOTELYSAVE_WEBSITE = process.env.REMOTELYSAVE_WEBSITE || "";
|
||||||
|
const DEFAULT_REMOTELYSAVE_CLIENT_ID = process.env.REMOTELYSAVE_CLIENT_ID || "";
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
entry: "./src/main.ts",
|
entry: "./src/main.ts",
|
||||||
@ -20,6 +22,8 @@ module.exports = {
|
|||||||
"process.env.DEFAULT_DROPBOX_APP_KEY": `"${DEFAULT_DROPBOX_APP_KEY}"`,
|
"process.env.DEFAULT_DROPBOX_APP_KEY": `"${DEFAULT_DROPBOX_APP_KEY}"`,
|
||||||
"process.env.DEFAULT_ONEDRIVE_CLIENT_ID": `"${DEFAULT_ONEDRIVE_CLIENT_ID}"`,
|
"process.env.DEFAULT_ONEDRIVE_CLIENT_ID": `"${DEFAULT_ONEDRIVE_CLIENT_ID}"`,
|
||||||
"process.env.DEFAULT_ONEDRIVE_AUTHORITY": `"${DEFAULT_ONEDRIVE_AUTHORITY}"`,
|
"process.env.DEFAULT_ONEDRIVE_AUTHORITY": `"${DEFAULT_ONEDRIVE_AUTHORITY}"`,
|
||||||
|
"process.env.DEFAULT_REMOTELYSAVE_WEBSITE": `"${DEFAULT_REMOTELYSAVE_WEBSITE}"`,
|
||||||
|
"process.env.DEFAULT_REMOTELYSAVE_CLIENT_ID": `"${DEFAULT_REMOTELYSAVE_CLIENT_ID}"`,
|
||||||
}),
|
}),
|
||||||
// Work around for Buffer is undefined:
|
// Work around for Buffer is undefined:
|
||||||
// https://github.com/webpack/changelog-v5/issues/10
|
// https://github.com/webpack/changelog-v5/issues/10
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user