From ecee0b6236b0cd8bfda063a8dd288a49dfc82ece Mon Sep 17 00:00:00 2001
From: harytfw
Date: Sat, 7 Jan 2023 22:08:29 +0800
Subject: [PATCH 01/21] chore: refactor build process
Signed-off-by: harytfw
---
Makefile | 55 +++++++------------
scripts/gen_manifest.js | 11 +---
rollup.config.js => scripts/rollup.config.js | 40 +++++++++-----
.../test_bootstrap.js | 5 +-
scripts/utils.js | 17 ++++++
5 files changed, 67 insertions(+), 61 deletions(-)
rename rollup.config.js => scripts/rollup.config.js (63%)
rename test/bootstrap.mjs => scripts/test_bootstrap.js (90%)
create mode 100644 scripts/utils.js
diff --git a/Makefile b/Makefile
index 77069be..bb97034 100644
--- a/Makefile
+++ b/Makefile
@@ -1,46 +1,41 @@
export BUILD_VERSION ?= 2.1.4
-export SRC ?= $(shell realpath ./src)
export BUILD_DIR ?= $(shell realpath ./build)
-export BROWSER_LOCATION ?= $(shell which firefox-developer-edition)
-export NODE_MODULES ?= $(shell realpath ./node_modules)
+export SRC ?= $(shell realpath ./src)
export BUILD_PROFILE ?= debug
-export BUILD_COMMIT_ID ?= $(shell git rev-parse --short HEAD)
-export BUILD_DATE ?= $(shell date --rfc-3339=seconds)
-export BUILD_NODE_VERSION ?= $(shell node --version)
-export BUILD_ROLLUP_VERSION ?= $(shell npx rollup --version)
-export BUILD_OS ?= $(shell node -e "console.log(require('os').platform())")
export BUILD_WEBSOCKET_SERVER =
-
-export TARGET_BROWSER = firefox
-export TARGET_DIST = $(BUILD_DIR)/$(TARGET_BROWSER)/dist
-
export ENTRY_POINTS = background content_scripts options components
+export TARGET_BROWSER = firefox
+export TARGET_DIST = $(BUILD_DIR)/firefox/dist
.PHONY: ext-firefox
ext-firefox: TARGET_BROWSER = firefox
-ext-firefox: setup-dist assets compile lint package
+ext-firefox: TARGET_DIST = $(BUILD_DIR)/firefox/dist
+ext-firefox: setup-dist assets compile lint
+ cd $(TARGET_DIST) && zip -r $(BUILD_DIR)/artifacts/glitterdrag-pro-$(BUILD_VERSION)-firefox.zip .
.PHONY: ext-chromium
-ext-chromium: TARGET_BROWSER = chromium
-ext-chromium: setup-dist assets compile package
+ext-test: TARGET_BROWSER = chromium
+ext-chromium: TARGET_DIST = $(BUILD_DIR)/chromium/dist
+ext-chromium: setup-dist assets compile
+ cd $(TARGET_DIST) && zip -r $(BUILD_DIR)/artifacts/glitterdrag-pro-$(BUILD_VERSION)-chromium.zip .
.PHONY: test
test: TARGET_BROWSER = firefox-test
test: ENTRY_POINTS += test
+test: TARGET_DIST = $(BUILD_DIR)/test/dist
test: BUILD_WEBSOCKET_SERVER = ws://localhost:8000
test: setup-dist assets mocha-assets compile
- @node test/bootstrap.mjs
+ @node scripts/test_bootstrap.js
.PHONY: open-mocha
-open-mocha: TARGET_BROWSER = firefox-test
+open-mocha: TARGET_DIST = $(BUILD_DIR)/firefox/dist
open-mocha: ENTRY_POINTS += test
open-mocha: setup-dist assets mocha-assets compile
web-ext run -v \
-s $(TARGET_DIST) \
- -f "$(BROWSER_LOCATION)"
+ -f "$(shell which firefox-developer-edition)"
.PHONY: setup-dist
-setup-dist: TARGET_DIST = $(BUILD_DIR)/$(TARGET_BROWSER)/dist
setup-dist:
$(shell [ ! -d $(TARGET_DIST) ] && mkdir -p $(TARGET_DIST)/)
@@ -50,7 +45,7 @@ lint:
.PHONY: compile
compile:
- @npx rollup -c;
+ @pnpm exec rollup -c "./scripts/rollup.config.js"
.PHONY: manifest
manifest:
@@ -63,29 +58,19 @@ assets: manifest
cp -f $(SRC)/options/options.html $(TARGET_DIST)/options/options.html
@mkdir -p $(TARGET_DIST)/res && \
- cp -r -f $(NODE_MODULES)/simpledotcss/simple.min.css $(TARGET_DIST)/res/simple.min.css
+ cp -r -f node_modules/simpledotcss/simple.min.css $(TARGET_DIST)/res/simple.min.css
@cp -r -f $(SRC)/icon/ $(SRC)/_locales/ $(TARGET_DIST)/
.PHONY: mocha-assets
mocha-assets:
@mkdir -p $(TARGET_DIST)/test && \
- cp $(NODE_MODULES)/mocha/mocha.js \
- $(NODE_MODULES)/mocha/mocha.css \
- $(NODE_MODULES)/chai/chai.js \
+ cp node_modules/mocha/mocha.js \
+ node_modules/mocha/mocha.css \
+ node_modules/chai/chai.js \
$(SRC)/test/mocha.html \
$(TARGET_DIST)/test
-.PHONY: package
-package:
- @mkdir -p $(BUILD_DIR)/artifacts
- @cd $(TARGET_DIST) && zip -r $(BUILD_DIR)/artifacts/glitterdrag-pro-$(BUILD_VERSION)-$(TARGET_BROWSER).zip .
-
.PHONY: clean
clean:
- @rm -rf $(BUILD_DIR)
-
-
-start-server: port ?= 8000
-start-server:
- @python3 -m http.server $(port)
+ @rm -rf $(BUILD_DIR)
\ No newline at end of file
diff --git a/scripts/gen_manifest.js b/scripts/gen_manifest.js
index 6c9600d..4b74b7b 100644
--- a/scripts/gen_manifest.js
+++ b/scripts/gen_manifest.js
@@ -1,13 +1,4 @@
-
-function mustEnv(key) {
- const value = process.env[key]
- if (!value || value.length === 0) {
- console.error("require environment: " + key)
- process.exit(1)
- }
- return value
-}
-
+import { mustEnv } from "./utils.js"
const version = mustEnv("BUILD_VERSION")
const targetBrowser = mustEnv("TARGET_BROWSER")
diff --git a/rollup.config.js b/scripts/rollup.config.js
similarity index 63%
rename from rollup.config.js
rename to scripts/rollup.config.js
index 36787da..3da5734 100644
--- a/rollup.config.js
+++ b/scripts/rollup.config.js
@@ -1,3 +1,5 @@
+import { mustEnv, readStdout } from "./utils.js"
+
import svelte from 'rollup-plugin-svelte';
import resolve from '@rollup/plugin-node-resolve';
import typescript from '@rollup/plugin-typescript';
@@ -5,23 +7,32 @@ import replace from '@rollup/plugin-replace';
import { terser } from 'rollup-plugin-terser';
import autoPreprocess from 'svelte-preprocess';
import commonjs from '@rollup/plugin-commonjs';
-import pathLib from 'path'
+import pathLib from 'node:path'
+import os from "node:os"
-function mustEnv(name) {
- const value = process.env[name]
- if (!value) {
- console.error("require env: " + name)
- process.exit(1)
- }
- return value
-}
+const BUILD_COMMIT_ID = readStdout("git rev-parse --short HEAD")
+const BUILD_DATE = readStdout("date --rfc-3339=seconds")
+const BUILD_NODE_VERSION = readStdout("node --version")
+const BUILD_ROLLUP_VERSION = readStdout("npx rollup --version")
+const BUILD_OS = os.platform()
+const BUILD_PROFILE = mustEnv("BUILD_PROFILE")
+const BUILD_WEBSOCKET_SERVER = mustEnv("BUILD_WEBSOCKET_SERVER")
-const isProd = mustEnv("BUILD_PROFILE").toLowerCase() === 'prod';
+const isProd = BUILD_PROFILE.toLowerCase() === 'prod';
+const sourceMap = !isProd
const src = mustEnv("SRC")
const dist = mustEnv("TARGET_DIST")
const entryPoints = mustEnv("ENTRY_POINTS").split(" ")
-const safeEnvVar = Object.fromEntries(Object.entries(process.env).filter((entry) => entry[0].startsWith("BUILD")))
+const safeEnvVar = {
+ BUILD_COMMIT_ID,
+ BUILD_DATE,
+ BUILD_NODE_VERSION,
+ BUILD_ROLLUP_VERSION,
+ BUILD_OS,
+ BUILD_PROFILE,
+ BUILD_WEBSOCKET_SERVER,
+}
const useCustomElement = ["components"]
@@ -38,7 +49,7 @@ function getPlugins(entrypoint) {
}),
replace({
__ENV: JSON.stringify(safeEnvVar),
- __BUILD_PROFILE: JSON.stringify(process.env.BUILD_PROFILE),
+ __BUILD_PROFILE: JSON.stringify(BUILD_PROFILE),
preventAssignment: true
}),
svelte({
@@ -58,15 +69,16 @@ function getPlugins(entrypoint) {
return plugins
}
+console.log("sourcemap: ", sourceMap)
+
const output = []
-console.log("sourcemap: ", !isProd)
for (const e of entryPoints) {
output.push({
external: ["chai", "mocha", "Mocha"],
input: pathLib.join(src, e, "main.ts"),
output: {
- sourcemap: !isProd,
+ sourcemap: sourceMap,
globals: {
"chai": "chai",
"mocha": "mocha",
diff --git a/test/bootstrap.mjs b/scripts/test_bootstrap.js
similarity index 90%
rename from test/bootstrap.mjs
rename to scripts/test_bootstrap.js
index ade0448..ffd3c86 100644
--- a/test/bootstrap.mjs
+++ b/scripts/test_bootstrap.js
@@ -1,16 +1,17 @@
import { WebSocketServer } from 'ws';
import { spawn } from 'node:child_process';
+import webExt from 'web-ext'
const TEST_TIMEOUT_MS = 1000 * 60 * 5
-const { BROWSER_LOCATION, TARGET_DIST, BUILD_WEBSOCKET_SERVER } = process.env
+const { TARGET_DIST, BUILD_WEBSOCKET_SERVER } = process.env
async function runBrowser(signal) {
return new Promise((resolve, reject) => {
- const proc = spawn("web-ext", ["run", "-f", BROWSER_LOCATION, "-s", TARGET_DIST], {
+ const proc = spawn("pnpm", ["exec", "web-ext", "run", "-f", "firefox-developer-edition" , "-s", TARGET_DIST], {
stdio: ["ignore", process.stdout, process.stderr],
signal: signal
})
diff --git a/scripts/utils.js b/scripts/utils.js
new file mode 100644
index 0000000..6cc58d3
--- /dev/null
+++ b/scripts/utils.js
@@ -0,0 +1,17 @@
+
+import { execSync } from 'node:child_process'
+
+export function mustEnv(key) {
+ const value = process.env[key]
+ if (!value || value.length === 0) {
+ console.error("require environment: " + key)
+ process.exit(1)
+ }
+ return value
+}
+
+
+export function readStdout(command) {
+ const result = execSync(command)
+ return result.toString("utf8")
+}
From e3a626ebd624f5390ce17a6176e995313b5f8063 Mon Sep 17 00:00:00 2001
From: harytfw
Date: Sat, 7 Jan 2023 22:31:03 +0800
Subject: [PATCH 02/21] chore: fix build
Signed-off-by: harytfw
---
Makefile | 2 ++
scripts/rollup.config.js | 2 +-
scripts/utils.js | 5 ++++-
3 files changed, 7 insertions(+), 2 deletions(-)
diff --git a/Makefile b/Makefile
index bb97034..cf1a5ff 100644
--- a/Makefile
+++ b/Makefile
@@ -7,6 +7,8 @@ export ENTRY_POINTS = background content_scripts options components
export TARGET_BROWSER = firefox
export TARGET_DIST = $(BUILD_DIR)/firefox/dist
+$(shell mkdir -p $(BUILD_DIR)/artifacts/)
+
.PHONY: ext-firefox
ext-firefox: TARGET_BROWSER = firefox
ext-firefox: TARGET_DIST = $(BUILD_DIR)/firefox/dist
diff --git a/scripts/rollup.config.js b/scripts/rollup.config.js
index 3da5734..560c9b8 100644
--- a/scripts/rollup.config.js
+++ b/scripts/rollup.config.js
@@ -16,7 +16,7 @@ const BUILD_NODE_VERSION = readStdout("node --version")
const BUILD_ROLLUP_VERSION = readStdout("npx rollup --version")
const BUILD_OS = os.platform()
const BUILD_PROFILE = mustEnv("BUILD_PROFILE")
-const BUILD_WEBSOCKET_SERVER = mustEnv("BUILD_WEBSOCKET_SERVER")
+const BUILD_WEBSOCKET_SERVER = mustEnv("BUILD_WEBSOCKET_SERVER", "")
const isProd = BUILD_PROFILE.toLowerCase() === 'prod';
const sourceMap = !isProd
diff --git a/scripts/utils.js b/scripts/utils.js
index 6cc58d3..ec7cab8 100644
--- a/scripts/utils.js
+++ b/scripts/utils.js
@@ -1,9 +1,12 @@
import { execSync } from 'node:child_process'
-export function mustEnv(key) {
+export function mustEnv(key, default_value) {
const value = process.env[key]
if (!value || value.length === 0) {
+ if (typeof default_value !== "undefined") {
+ return default_value
+ }
console.error("require environment: " + key)
process.exit(1)
}
From 1b46221ba26abd0cf25bbe217701053c0cd1f73d Mon Sep 17 00:00:00 2001
From: harytfw
Date: Sat, 7 Jan 2023 22:32:55 +0800
Subject: [PATCH 03/21] fix: fix check defaultPrevented at dragstart event
Signed-off-by: harytfw
---
src/content_scripts/drag.ts | 27 +++++++++++++++------------
src/content_scripts/main.ts | 2 +-
2 files changed, 16 insertions(+), 13 deletions(-)
diff --git a/src/content_scripts/drag.ts b/src/content_scripts/drag.ts
index 8b5b6b8..8b89ccb 100644
--- a/src/content_scripts/drag.ts
+++ b/src/content_scripts/drag.ts
@@ -46,15 +46,15 @@ export class DragController {
return
}
for (const n of ["dragstart", "dragover", "dragenter", 'dragleave', "drop", "dragend"]) {
- this.eventSource.addEventListener(n, this.handler, true)
- this.eventSource.addEventListener(n, this.handler, false)
+ this.eventSource.addEventListener(n, this.handler, {capture: true})
+ this.eventSource.addEventListener(n, this.handler, {capture: false})
}
}
stop() {
for (const n of ["dragstart", "dragover", "dragenter", 'dragleave', "drop", "dragend"]) {
- this.eventSource.removeEventListener(n, this.handler, true);
- this.eventSource.removeEventListener(n, this.handler, false);
+ this.eventSource.removeEventListener(n, this.handler, {capture: true});
+ this.eventSource.removeEventListener(n, this.handler, {capture: false});
}
}
@@ -169,14 +169,17 @@ export class DragController {
}
private checkDragStart(event: DragEvent) {
- if (event.eventPhase !== EventPhase.capturing) {
- return
- }
- this.sourceTarget = this.findSourceTarget(event)
- if (this.sourceTarget != null) {
- this.initFramePosition()
- const op = this.makeOp(OpType.start, this.sourceTarget, event)
- this.c.applyOp(op)
+ if (event.eventPhase === EventPhase.bubbling) {
+ if (!event.defaultPrevented) {
+ this.sourceTarget = this.findSourceTarget(event)
+ if (this.sourceTarget != null) {
+ this.initFramePosition()
+ const op = this.makeOp(OpType.start, this.sourceTarget, event)
+ this.c.applyOp(op)
+ }
+ } else {
+ log.V("host page cancel dragtart event")
+ }
}
}
diff --git a/src/content_scripts/main.ts b/src/content_scripts/main.ts
index 69f7634..ab88605 100644
--- a/src/content_scripts/main.ts
+++ b/src/content_scripts/main.ts
@@ -67,7 +67,7 @@ async function setup() {
const opExecutor = new OpExecutor()
opExecutor.reset()
- controller = new DragController(document.documentElement, opExecutor);
+ controller = new DragController(window, opExecutor);
onDocumentLoaded(async () => {
await browser.runtime.sendMessage(buildRuntimeMessage(RuntimeMessageName.contextScriptLoaded, null))
From 9ebe3e4c3b57a4c3a40944c3f22066d5cc33f131 Mon Sep 17 00:00:00 2001
From: harytfw
Date: Sun, 8 Jan 2023 12:47:07 +0800
Subject: [PATCH 04/21] docs: add limitation
Signed-off-by: harytfw
---
README.md | 9 +++++++++
1 file changed, 9 insertions(+)
diff --git a/README.md b/README.md
index 62777b2..5c88faa 100644
--- a/README.md
+++ b/README.md
@@ -25,3 +25,12 @@ make ext-chromium
```
After the build process completed, `./build/firefox/dist` contains compiled result, `./build/firefox/artifacts` contains package file.
+
+## Limitation
+
+The extension has limited support at these situation:
+
+- Can not work on privilege pages or restricted pages, like , ,
+- Can not work when dragging any selection from page A and dropping at page B, and page A and page B have different origin. (For example, [host page](www.example.com) is A, it embeds [youtube video](https://www.youtube.com/embed/-88qGXDmh3E) as B)
+- When host page explicitly uses drag-and-drop features, the extension will stop working temporarily
+- Unable show icon of menu on when [CSP](https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Content-Security-Policy/Sources#sources) of host page prohibits *data:* URLs
From 70c3b5bb4e0db6ba9d5e3d75fb5e32cc1c876b34 Mon Sep 17 00:00:00 2001
From: harytfw
Date: Sun, 8 Jan 2023 13:25:47 +0800
Subject: [PATCH 05/21] feat: support other protocol uri
Signed-off-by: harytfw
---
src/utils/url.ts | 25 ++++++++++++++++++-------
1 file changed, 18 insertions(+), 7 deletions(-)
diff --git a/src/utils/url.ts b/src/utils/url.ts
index 3dc3f84..4446585 100644
--- a/src/utils/url.ts
+++ b/src/utils/url.ts
@@ -28,18 +28,18 @@ function matchRe(s: string, pat: RegExp): boolean {
return r && r.length > 0
}
-function isTopLevelDomain(url: URL): boolean {
- return parse(url.hostname).isIcann
-}
-
const log = rootLog.subLogger(LogLevel.VVV, "urlFixer")
const urlLRU = new TinyLRU()
export class URLFixer {
private protocol = "https://"
+ private specialProtocols: Set = new Set()
- constructor() {
+ constructor(extraProtocols?: string[]) {
+ if (Array.isArray(extraProtocols)) {
+ this.specialProtocols = new Set(extraProtocols)
+ }
}
fix(urlLike?: string): URL | null {
@@ -52,7 +52,13 @@ export class URLFixer {
return res
}
- internalFix(urlLike?: string): URL | null {
+ private isTopLevelDomain(url: URL): boolean {
+ if (parse(url.hostname).isIcann) {
+ return true
+ }
+ }
+
+ private internalFix(urlLike?: string): URL | null {
log.VVV("try fix url: ", urlLike)
@@ -97,7 +103,12 @@ export class URLFixer {
const url = this.tryBuildURL(urlLike)
if (url) {
- if (isTopLevelDomain(url)) {
+ if (this.isTopLevelDomain(url)) {
+ log.VVV(urlLike, "is top level domain")
+ return url
+ }
+ if (this.specialProtocols.has(url.protocol)) {
+ log.VVV(urlLike, "has special protocol", url.protocol)
return url
}
return null
From 24ed7e6603c664eea101364ce9ef72249084f885 Mon Sep 17 00:00:00 2001
From: harytfw
Date: Sun, 8 Jan 2023 13:26:26 +0800
Subject: [PATCH 06/21] test: fix usage of assert
Signed-off-by: harytfw
---
src/context/context.test.ts | 17 ++++++++---------
src/utils/test_helper.ts | 12 ++++--------
src/utils/url.test.ts | 30 ++++++++++++++++++++++--------
src/utils/var_substitute.test.ts | 6 +++---
4 files changed, 37 insertions(+), 28 deletions(-)
diff --git a/src/context/context.test.ts b/src/context/context.test.ts
index 85c2be9..c8b914e 100644
--- a/src/context/context.test.ts
+++ b/src/context/context.test.ts
@@ -1,21 +1,20 @@
import { blankExecuteContext } from "./test_helper"
import { ContextType } from "../config/config"
-import { assertEqual } from "../utils/test_helper"
import { primaryContextType } from "./utils"
-
+import { assert } from 'chai'
describe("test context", () => {
- it("primary type",async () => {
- let ctx = await blankExecuteContext()
+ it("primary type", async () => {
+ let ctx = await blankExecuteContext()
ctx.data.selection = "hello"
- assertEqual(primaryContextType(ctx), ContextType.selection)
+ assert.deepEqual(primaryContextType(ctx), ContextType.selection)
- ctx = await blankExecuteContext()
+ ctx = await blankExecuteContext()
ctx.data.link = "http://example.com/"
- assertEqual(primaryContextType(ctx), ContextType.link)
+ assert.deepEqual(primaryContextType(ctx), ContextType.link)
- ctx = await blankExecuteContext()
+ ctx = await blankExecuteContext()
ctx.data.imageSource = "http://example.com/a.jpg"
- assertEqual(primaryContextType(ctx), ContextType.image)
+ assert.deepEqual(primaryContextType(ctx), ContextType.image)
})
})
\ No newline at end of file
diff --git a/src/utils/test_helper.ts b/src/utils/test_helper.ts
index 091cbb8..64906c3 100644
--- a/src/utils/test_helper.ts
+++ b/src/utils/test_helper.ts
@@ -1,21 +1,17 @@
import browser from 'webextension-polyfill';
-import isEqual from "lodash-es/isEqual"
+import { assert } from 'chai'
export function assertEqual(expected: any, actual: any, ...message: any[]) {
- return assertOk(isEqual(actual, expected), ...message)
+ return assert.deepEqual(actual, expected, ...message)
}
export function assertOk(value: any, ...message: any[]) {
- if (!value) {
- throw new Error(message.join(" "))
- }
+ return assert.ok(value, ...message)
}
export function assertFail(value: any, ...message: any[]) {
- if (!!value) {
- throw new Error(message.join(" "))
- }
+ return assert.notOk(value, ...message)
}
export async function closeTab(tabIds: number | number[], delayMs?: number) {
diff --git a/src/utils/url.test.ts b/src/utils/url.test.ts
index d288ae2..42d5d7b 100644
--- a/src/utils/url.test.ts
+++ b/src/utils/url.test.ts
@@ -1,22 +1,23 @@
import { URLFixer } from './url'
-import { assertEqual, assertFail, assertOk } from './test_helper';
import { rootLog } from './log';
+import { assert } from 'chai'
describe('test url fixer', () => {
- const fixer = new URLFixer()
it('good url', () => {
+ const fixer = new URLFixer()
const urls = [
"http://example.com/",
"http://example.org/",
"http://mirrors.tuna.tsinghua.edu.cn/anaconda/pkgs/main/win-64"
]
for (const c of urls) {
- assertEqual(fixer.fix(c).toString(), c)
+ assert.deepEqual(fixer.fix(c).toString(), c)
}
})
it('bad url', () => {
+ const fixer = new URLFixer()
const urls = [
"12345",
@@ -24,35 +25,38 @@ describe('test url fixer', () => {
]
for (const c of urls) {
- assertFail(fixer.fix(c), c)
+ assert.notOk(fixer.fix(c), c)
}
});
it('valid ipv4', () => {
+ const fixer = new URLFixer()
const urls = [
"8.8.8.8",
"192.168.1.1",
"127.0.0.1"
]
for (const c of urls) {
- assertOk(fixer.fix(c), c)
+ assert.ok(fixer.fix(c), c)
}
});
it('valid ipv6', () => {
+ const fixer = new URLFixer()
const urls = [
"2001:db8:0:0:1:0:0:1",
"2001:db8::1:0:0:1",
"::1"
]
for (const c of urls) {
- assertFail(fixer.fix(c), "not support plain ipv6: ", c)
- assertOk(fixer.fix("[" + c + "]"), "ipv6 url: ", c)
+ assert.notOk(fixer.fix(c), "not support plain ipv6: " + c)
+ assert.ok(fixer.fix("[" + c + "]"), "ipv6 url: " + c)
}
});
it('fix url', () => {
+ const fixer = new URLFixer()
const urls = [
[
"www.example.com/a",
@@ -79,7 +83,17 @@ describe('test url fixer', () => {
for (const c of urls) {
const fixed = fixer.fix(c[0]).toString()
rootLog.VVV("origin: ", c[0], "fixed:", fixed, " expected: ", c[1], " same: ", fixed == c[1])
- assertEqual(fixed, c[1], "expected: ", c[1], "but got:", fixed)
+ assert.deepEqual(fixed, c[1])
}
});
+ it('special protocol url', function () {
+ const fixer = new URLFixer(["magnet:", "mailto:"])
+ const urls = [
+ "magnet:?xt=urn:btih:000000000000000000000000000&dn",
+ "mailto:someone@example.com"
+ ]
+ for (const url of urls) {
+ assert.ok(fixer.fix(url))
+ }
+ })
});
\ No newline at end of file
diff --git a/src/utils/var_substitute.test.ts b/src/utils/var_substitute.test.ts
index e0439a3..d0ccb5c 100644
--- a/src/utils/var_substitute.test.ts
+++ b/src/utils/var_substitute.test.ts
@@ -1,6 +1,6 @@
import { assertEqual } from "./test_helper"
import { VarSubstituteTemplate } from "./var_substitute"
-
+import { assert } from 'chai'
describe('test variable substitute', () => {
@@ -20,7 +20,7 @@ describe('test variable substitute', () => {
for (const c of cases) {
const template = new VarSubstituteTemplate(c)
const result = template.substitute(m)
- assertEqual(result, c, "expected: ", c, ", actually: ", result)
+ assert.deepEqual(result, c)
}
})
@@ -52,7 +52,7 @@ describe('test variable substitute', () => {
for (const [s, expected] of cases) {
const template = new VarSubstituteTemplate(s)
const result = template.substitute(m)
- assertEqual(result, expected, "expected: ", expected, ", actually: ", result)
+ assert.deepEqual(result, expected)
}
})
})
\ No newline at end of file
From f3ed0fc1e85782ce1fd34f45dc1c3796ef8c265b Mon Sep 17 00:00:00 2001
From: harytfw
Date: Sun, 8 Jan 2023 14:12:57 +0800
Subject: [PATCH 07/21] feat: add configuration for protocol of smartURL
Signed-off-by: harytfw
---
src/config/config.ts | 13 +++++++++++--
src/content_scripts/op_source.ts | 16 ++++++++++++++--
2 files changed, 25 insertions(+), 4 deletions(-)
diff --git a/src/config/config.ts b/src/config/config.ts
index 7d44da3..5dc9484 100644
--- a/src/config/config.ts
+++ b/src/config/config.ts
@@ -107,13 +107,21 @@ export interface PlainLogConfig {
}
export class SmartURLConfig {
- cfg: KVRecord
- constructor(cfg: KVRecord) {
+ cfg: PlainSmartURLConfig
+
+ constructor(cfg: PlainSmartURLConfig) {
this.cfg = cfg
}
+
+ get protocols(): string[] {
+ return defaultTo(this.cfg.protocols, [])
+ }
}
+export interface PlainSmartURLConfig {
+ protocols?: string[]
+}
export class ModeConfig {
@@ -653,6 +661,7 @@ export interface PlainConfiguration {
assets?: PlainAsset[],
scripts?: PlainScript[],
compatibility?: PlainCompatibilityRule[],
+ smartURL?: PlainSmartURLConfig,
}
export class BroadcastEventTarget {
diff --git a/src/content_scripts/op_source.ts b/src/content_scripts/op_source.ts
index 99eb16d..050d900 100644
--- a/src/content_scripts/op_source.ts
+++ b/src/content_scripts/op_source.ts
@@ -1,8 +1,15 @@
-import { ContextType } from "../config/config"
+import { configBroadcast, ContextType, type ReadonlyConfiguration } from "../config/config"
import type { KVRecord } from "../types"
import { URLFixer } from "../utils/url"
import type { OpSummary } from "./op"
+
+let config: ReadonlyConfiguration | null = null
+
+configBroadcast.addListener((cfg) => {
+ config = cfg
+})
+
export class OpSource {
private _isContentEditable: boolean = false
private _isDraggable: boolean = false
@@ -72,7 +79,12 @@ export class OpSource {
}
if (os.type === ContextType.selection) {
- const url = new URLFixer().fix(os.text)
+ let protocols = []
+ if (config) {
+ protocols = Array.from(config.smartURL.protocols)
+ }
+ const fixer = new URLFixer(protocols)
+ const url = fixer.fix(os.text)
if (url) {
os._link = url.toString()
os._linkText = os.text
From 79ea12a601f3f62cd45b630ad41265af3e58f05e Mon Sep 17 00:00:00 2001
From: harytfw
Date: Sun, 8 Jan 2023 14:30:07 +0800
Subject: [PATCH 08/21] refactor: replace KVRecord with concrete type
Signed-off-by: harytfw
---
src/background/executor.test.ts | 4 +-
src/config/config.ts | 183 ++++++++++++++------------------
2 files changed, 84 insertions(+), 103 deletions(-)
diff --git a/src/background/executor.test.ts b/src/background/executor.test.ts
index 2e07e20..e84b3b6 100644
--- a/src/background/executor.test.ts
+++ b/src/background/executor.test.ts
@@ -29,14 +29,14 @@ describe("test executor", async () => {
})
it("copy text", async () => {
- const ctx = await blankExecuteContext(new ActionConfig({ "command": "copy" }))
+ const ctx = await blankExecuteContext(new ActionConfig({ command: CommandKind.copy }))
ctx.data.selection = "copy demo"
await executor.copyHandler(ctx)
})
it("copy image", async () => {
- const ctx = await blankExecuteContext(new ActionConfig({ "command": "copy" }))
+ const ctx = await blankExecuteContext(new ActionConfig({ command: CommandKind.copy }))
ctx.data.imageSource = ""
await executor.copyHandler(ctx)
diff --git a/src/config/config.ts b/src/config/config.ts
index 5dc9484..7bd4a7a 100644
--- a/src/config/config.ts
+++ b/src/config/config.ts
@@ -91,19 +91,19 @@ export enum Feature {
export class LogConfig {
- private cfg: KVRecord
+ private cfg: PlainLogConfig
- constructor(cfg: KVRecord) {
+ constructor(cfg: PlainLogConfig) {
this.cfg = cfg
}
get level(): LogLevel {
- return defaultTo(this.cfg['level'], LogLevel.S)
+ return defaultTo(this.cfg.level, LogLevel.S)
}
}
export interface PlainLogConfig {
- level: number
+ level?: number
}
export class SmartURLConfig {
@@ -125,66 +125,56 @@ export interface PlainSmartURLConfig {
export class ModeConfig {
- private cfg: KVRecord
+ private cfg: PlainModeConfig
- constructor(cfg: KVRecord) {
+ constructor(cfg: PlainModeConfig) {
this.cfg = cfg
}
get link(): OperationMode {
- return defaultTo(this.cfg['link'], OperationMode.normal)
+ return defaultTo(this.cfg.link, OperationMode.normal)
}
get selection(): OperationMode {
- return defaultTo(this.cfg['selection'], OperationMode.normal)
+ return defaultTo(this.cfg.selection, OperationMode.normal)
}
get image(): OperationMode {
- return defaultTo(this.cfg['image'], OperationMode.normal)
+ return defaultTo(this.cfg.image, OperationMode.normal)
}
}
export interface PlainModeConfig {
- link?: string
- selection?: string
- image?: string
+ link?: OperationMode
+ selection?: OperationMode
+ image?: OperationMode
}
export class CommonConfig {
- private cfg: KVRecord
+ private cfg: PlainCommonConfig
private _mode: ModeConfig
- constructor(cfg: KVRecord) {
+ constructor(cfg: PlainCommonConfig) {
this.cfg = cfg
- this._mode = new ModeConfig(defaultTo(this.cfg['mode'], {}))
+ this._mode = new ModeConfig(defaultTo(this.cfg.mode, {}))
}
get mode(): ModeConfig {
return this._mode
}
- get allow(): string[] {
- return defaultTo(this.cfg['allow'], [])
- }
-
- get disallow(): string[] {
- return defaultTo(this.cfg['disallow'], [])
- }
-
get minDistance(): number {
- return defaultTo(this.cfg['minDistance'], 0)
+ return defaultTo(this.cfg.minDistance, 0)
}
get maxDistance(): number {
- return defaultTo(this.cfg['maxDistance'], 0)
+ return defaultTo(this.cfg.maxDistance, 0)
}
}
export interface PlainCommonConfig {
mode?: PlainModeConfig
- allowHost?: string[]
- disallowHost?: string[]
minDistance?: number
maxDistance?: number
}
@@ -195,9 +185,9 @@ export enum MenuLayout {
}
export class MenuConfig {
- cfg: KVRecord
+ cfg: PlainMenuConfig
- constructor(cfg: KVRecord) {
+ constructor(cfg: PlainMenuConfig) {
this.cfg = cfg
}
@@ -207,19 +197,11 @@ export interface PlainMenuConfig {
}
-export class ListModeConfig {
- cfg: KVRecord
-
- constructor(cfg: KVRecord) {
- this.cfg = cfg
- }
-}
-
export class Script {
- private cfg: KVRecord
+ private cfg: PlainScript
- constructor(cfg: KVRecord) {
+ constructor(cfg: PlainScript) {
this.cfg = cfg
}
@@ -231,15 +213,15 @@ export class Script {
}
get id() {
- return defaultTo(this.cfg['id'], "")
+ return defaultTo(this.cfg.id, "")
}
get text() {
- return defaultTo(this.cfg['text'], "")
+ return defaultTo(this.cfg.text, "")
}
get name() {
- return defaultTo(this.cfg['name'], this.id)
+ return defaultTo(this.cfg.name, this.id)
}
}
@@ -256,14 +238,14 @@ export class CommandRequest {
private _url: string
private _query: KVRecord
private _protocol: string
- private cfg: KVRecord
+ private cfg: PlainCommandRequest
constructor(cfg: PlainCommandRequest) {
this.cfg = cloneDeep(cfg)
- const urlObj = new URL(cfg['url'])
+ const urlObj = new URL(cfg.url)
- this._query = defaultTo(cfg['query'], {})
+ this._query = defaultTo(cfg.query, {})
for (const [k, v] of urlObj.searchParams) {
this._query[k] = cloneDeep(v)
}
@@ -289,11 +271,11 @@ export class CommandRequest {
}
get id() {
- return defaultTo(this.cfg['id'], '')
+ return defaultTo(this.cfg.id, '')
}
get name() {
- return defaultTo(this.cfg["name"], this.id)
+ return defaultTo(this.cfg.name, this.id)
}
get method() {
@@ -331,41 +313,41 @@ export const enum AssetType {
}
export class Asset {
- private cfg: KVRecord
+ private cfg: PlainAsset
- constructor(cfg: KVRecord) {
+ constructor(cfg: PlainAsset) {
this.cfg = cfg
}
get id(): string {
- return defaultTo(this.cfg["id"], "")
+ return defaultTo(this.cfg.id, "")
}
get name(): string {
- return defaultTo(this.cfg["name"], this.id)
+ return defaultTo(this.cfg.name, this.id)
}
get type(): AssetType {
- return defaultTo(this.cfg["type"], AssetType.html)
+ return defaultTo(this.cfg.type, AssetType.html)
}
get data(): string {
- return defaultTo(this.cfg["data"], "")
+ return defaultTo(this.cfg.data, "")
}
}
export interface PlainAsset {
id?: string
name?: string
- type?: string
+ type?: AssetType
data?: string
}
export class CommandConfig {
- private cfg: KVRecord
+ private cfg: PlainCommandConfig
- constructor(cfg: KVRecord) {
+ constructor(cfg: PlainCommandConfig) {
this.cfg = cfg
}
@@ -383,53 +365,53 @@ export class CommandConfig {
}
get activeTab(): boolean {
- return defaultTo(this.cfg['activeTab'], true)
+ return defaultTo(this.cfg.activeTab, true)
}
get tabPosition(): TabPosition {
- return defaultTo(this.cfg['tabPosition'], TabPosition.next)
+ return defaultTo(this.cfg.tabPosition, TabPosition.next)
}
get requestId(): string {
- return defaultTo(this.cfg['requestId'], "")
+ return defaultTo(this.cfg.requestId, "")
}
get scriptId(): string {
- return defaultTo(this.cfg['scriptId'], "")
+ return defaultTo(this.cfg.scriptId, "")
}
get directory(): string {
- return defaultTo(this.cfg['directory'], "")
+ return defaultTo(this.cfg.directory, "")
}
get showSaveAsDialog(): boolean {
- return defaultTo(this.cfg['showSaveAsDialog'], false)
+ return defaultTo(this.cfg.showSaveAsDialog, false)
}
get preferDataTypes(): ContextDataType[] {
- return defaultTo(this.cfg['preferDataTypes'], [])
+ return defaultTo(this.cfg.preferDataTypes, [])
}
get container(): string {
- return defaultTo(this.cfg['container'], "")
+ return defaultTo(this.cfg.container, "")
}
}
export interface PlainCommandConfig {
activeTab?: boolean
- tabPosition?: string
+ tabPosition?: TabPosition
requestId?: string
scriptId?: string
directory?: string
showSaveAsDialog?: boolean
- preferDataTypes?: string[]
+ preferDataTypes?: ContextDataType[]
container?: string
}
export class ConditionConfig {
- private cfg: KVRecord
+ private cfg: PlainConditionConfig
- constructor(cfg: KVRecord) {
+ constructor(cfg: PlainConditionConfig) {
this.cfg = cfg
}
@@ -442,39 +424,39 @@ export class ConditionConfig {
}
get modes(): OperationMode[] {
- return defaultTo(this.cfg['modes'], [])
+ return defaultTo(this.cfg.modes, [])
}
get contextTypes(): ContextType[] {
- return defaultTo(this.cfg['contextTypes'], [])
+ return defaultTo(this.cfg.contextTypes, [])
}
get directions(): Direction[] {
- return defaultTo(this.cfg['directions'], [])
+ return defaultTo(this.cfg.directions, [])
}
get extra(): ContextType[] {
- return defaultTo(this.cfg['extra'], [])
+ return defaultTo(this.cfg.extra, [])
}
}
export interface PlainConditionConfig {
- modes?: string[]
- contextTypes?: string[],
- directions?: string[],
- extra?: string[]
+ modes?: OperationMode[]
+ contextTypes?: ContextType[],
+ directions?: Direction[],
+ extra?: ContextType[]
}
export class ActionConfig {
- private cfg: KVRecord
+ private cfg: PlainActionConfig
private _conditionConfig: ConditionConfig
private _config: CommandConfig
- constructor(cfg: KVRecord) {
+ constructor(cfg: PlainActionConfig) {
this.cfg = cfg
- this._conditionConfig = new ConditionConfig(defaultTo(cfg['condition'], {}))
- this._config = new CommandConfig(defaultTo(cfg['config'], {}))
+ this._conditionConfig = new ConditionConfig(defaultTo(cfg.condition, {}))
+ this._config = new CommandConfig(defaultTo(cfg.config, {}))
}
toPlainObject(): PlainActionConfig {
@@ -490,15 +472,15 @@ export class ActionConfig {
}
get id(): string {
- return defaultTo(this.cfg['id'], "")
+ return defaultTo(this.cfg.id, "")
}
get name(): string {
- return defaultTo(this.cfg['name'], this.id)
+ return defaultTo(this.cfg.name, this.id)
}
get prompt(): string {
- return defaultTo(this.cfg['prompt'], "")
+ return defaultTo(this.cfg.prompt, "")
}
get condition(): Readonly {
@@ -506,7 +488,7 @@ export class ActionConfig {
}
get command(): CommandKind {
- return defaultTo(this.cfg['command'], CommandKind.invalid)
+ return defaultTo(this.cfg.command, CommandKind.invalid)
}
get config(): CommandConfig {
@@ -514,11 +496,11 @@ export class ActionConfig {
}
get icon(): string {
- return defaultTo(this.cfg['icon'], "")
+ return defaultTo(this.cfg.icon, "")
}
get iconAssetId(): string {
- return defaultTo(this.cfg['iconAssetId'], "")
+ return defaultTo(this.cfg.iconAssetId, "")
}
}
@@ -526,7 +508,7 @@ export interface PlainActionConfig {
id?: string
name?: string
condition?: PlainConditionConfig
- command?: string
+ command?: CommandKind
config?: PlainCommandConfig
icon?: string
iconAssetId?: string
@@ -544,13 +526,11 @@ export interface PlainCompatibilityRule { host?: string, regexp?: string, status
export class CompatibilityRule {
- private cfg: KVRecord
private _host: string
private _regexp: string
private _status: CompatibilityStatus
constructor(cfg: PlainCompatibilityRule) {
- this.cfg = cfg
this._host = defaultTo(cfg.host, "")
this._regexp = defaultTo(cfg.regexp, "")
this._status = defaultTo(cfg.status, CompatibilityStatus.enable)
@@ -590,19 +570,19 @@ export class Configuration {
private _scripts: Script[]
private _compatibility: CompatibilityRule[]
- constructor(data?: KVRecord) {
- data = defaultTo(data, {})
+ constructor(cfg?: PlainConfiguration) {
+ cfg = defaultTo(cfg, {})
- this._log = new LogConfig(defaultTo(data['log'], {}))
- this._smartURL = new SmartURLConfig(defaultTo(data['smartURL'], {}))
- this._common = new CommonConfig(defaultTo(data['common'], {}))
- this._menu = new MenuConfig(defaultTo(data['grid'], {}))
- this._actions = defaultTo(data['actions'], []).map((c: KVRecord) => new ActionConfig(c))
- this._features = new Set(defaultTo(data['features'], []))
- this._requests = defaultTo(data['requests'], []).map((r: KVRecord) => new CommandRequest(r))
- this._assets = defaultTo(data['assets'], []).map((r: KVRecord) => new Asset(r))
- this._scripts = defaultTo(data['scripts'], []).map((r: KVRecord) => new Script(r))
- this._compatibility = defaultTo(data["compatibility"], []).map((r: KVRecord) => new CompatibilityRule(r))
+ this._log = new LogConfig(defaultTo(cfg.log, {}))
+ this._smartURL = new SmartURLConfig(defaultTo(cfg.smartURL, {}))
+ this._common = new CommonConfig(defaultTo(cfg.common, {}))
+ this._menu = new MenuConfig(defaultTo(cfg.menu, {}))
+ this._actions = defaultTo(cfg.actions, []).map((c: PlainActionConfig) => new ActionConfig(c))
+ this._requests = defaultTo(cfg.requests, []).map((r: PlainCommandRequest) => new CommandRequest(r))
+ this._assets = defaultTo(cfg.assets, []).map((r: PlainAsset) => new Asset(r))
+ this._scripts = defaultTo(cfg.scripts, []).map((r: PlainScript) => new Script(r))
+ this._compatibility = defaultTo(cfg.compatibility, []).map((r: PlainCompatibilityRule) => new CompatibilityRule(r))
+ this._features = new Set(defaultTo(cfg.features as Feature[], []))
}
get features(): Set {
@@ -662,6 +642,7 @@ export interface PlainConfiguration {
scripts?: PlainScript[],
compatibility?: PlainCompatibilityRule[],
smartURL?: PlainSmartURLConfig,
+ features?: string[]
}
export class BroadcastEventTarget {
From 55ee756af717a6998b5bed85ce6c64ec9478d8a2 Mon Sep 17 00:00:00 2001
From: harytfw
Date: Tue, 10 Jan 2023 14:30:07 +0800
Subject: [PATCH 09/21] chore: refactor build cli
Signed-off-by: harytfw
---
Makefile | 90 +++-------
package.json | 8 +-
pnpm-lock.yaml | 83 +++++----
scripts/build_rollup.mjs | 82 +++++++++
scripts/cli.mjs | 295 ++++++++++++++++++++++++++++++++
scripts/gen_manifest.js | 119 -------------
scripts/gen_manifest.mjs | 124 ++++++++++++++
scripts/rollup.config.js | 94 ----------
scripts/test_bootstrap.js | 95 ----------
scripts/{utils.js => utils.mjs} | 4 +
src/build_info.ts | 10 +-
src/global.d.ts | 12 +-
src/test/mocha_init.ts | 2 +-
13 files changed, 598 insertions(+), 420 deletions(-)
create mode 100644 scripts/build_rollup.mjs
create mode 100644 scripts/cli.mjs
delete mode 100644 scripts/gen_manifest.js
create mode 100644 scripts/gen_manifest.mjs
delete mode 100644 scripts/rollup.config.js
delete mode 100644 scripts/test_bootstrap.js
rename scripts/{utils.js => utils.mjs} (85%)
diff --git a/Makefile b/Makefile
index cf1a5ff..40d6f81 100644
--- a/Makefile
+++ b/Makefile
@@ -1,78 +1,38 @@
export BUILD_VERSION ?= 2.1.4
export BUILD_DIR ?= $(shell realpath ./build)
-export SRC ?= $(shell realpath ./src)
-export BUILD_PROFILE ?= debug
-export BUILD_WEBSOCKET_SERVER =
-export ENTRY_POINTS = background content_scripts options components
-export TARGET_BROWSER = firefox
-export TARGET_DIST = $(BUILD_DIR)/firefox/dist
+export BUILD_SRC ?= $(shell realpath ./src)
-$(shell mkdir -p $(BUILD_DIR)/artifacts/)
+cli = node ./scripts/cli.mjs
+.PHONY: build
+build: TARGET ?= firefox
+build:
+ $(cli) build --target $(TARGET)
+
+.PHONY: build-watch
+build-watch: TARGET ?= firefox
+build-watch:
+ $(cli) build --target $(TARGET) --watch
.PHONY: ext-firefox
-ext-firefox: TARGET_BROWSER = firefox
-ext-firefox: TARGET_DIST = $(BUILD_DIR)/firefox/dist
-ext-firefox: setup-dist assets compile lint
- cd $(TARGET_DIST) && zip -r $(BUILD_DIR)/artifacts/glitterdrag-pro-$(BUILD_VERSION)-firefox.zip .
+ext-firefox: BUILD_PROFILE ?= debug
+ext-firefox:
+ $(cli) build --target firefox --lint --profile $(BUILD_PROFILE)
.PHONY: ext-chromium
-ext-test: TARGET_BROWSER = chromium
-ext-chromium: TARGET_DIST = $(BUILD_DIR)/chromium/dist
-ext-chromium: setup-dist assets compile
- cd $(TARGET_DIST) && zip -r $(BUILD_DIR)/artifacts/glitterdrag-pro-$(BUILD_VERSION)-chromium.zip .
+ext-chromium: BUILD_PROFILE ?= debug
+ext-chromium:
+ $(cli) build --target chromium --profile $(BUILD_PROFILE)
.PHONY: test
-test: TARGET_BROWSER = firefox-test
-test: ENTRY_POINTS += test
-test: TARGET_DIST = $(BUILD_DIR)/test/dist
-test: BUILD_WEBSOCKET_SERVER = ws://localhost:8000
-test: setup-dist assets mocha-assets compile
- @node scripts/test_bootstrap.js
-
-.PHONY: open-mocha
-open-mocha: TARGET_DIST = $(BUILD_DIR)/firefox/dist
-open-mocha: ENTRY_POINTS += test
-open-mocha: setup-dist assets mocha-assets compile
- web-ext run -v \
- -s $(TARGET_DIST) \
- -f "$(shell which firefox-developer-edition)"
-
-.PHONY: setup-dist
-setup-dist:
- $(shell [ ! -d $(TARGET_DIST) ] && mkdir -p $(TARGET_DIST)/)
-
-.PHONY: lint
-lint:
- @pnpm exec web-ext lint -s $(TARGET_DIST)
-
-.PHONY: compile
-compile:
- @pnpm exec rollup -c "./scripts/rollup.config.js"
-
-.PHONY: manifest
-manifest:
- @node scripts/gen_manifest.js > $(TARGET_DIST)/manifest.json
-
-.PHONY: assets
-assets: manifest
-
- @mkdir -p $(TARGET_DIST)/options && \
- cp -f $(SRC)/options/options.html $(TARGET_DIST)/options/options.html
-
- @mkdir -p $(TARGET_DIST)/res && \
- cp -r -f node_modules/simpledotcss/simple.min.css $(TARGET_DIST)/res/simple.min.css
-
- @cp -r -f $(SRC)/icon/ $(SRC)/_locales/ $(TARGET_DIST)/
+test: TARGET ?= firefox
+test:
+ $(cli) test --target $(TARGET)
-.PHONY: mocha-assets
-mocha-assets:
- @mkdir -p $(TARGET_DIST)/test && \
- cp node_modules/mocha/mocha.js \
- node_modules/mocha/mocha.css \
- node_modules/chai/chai.js \
- $(SRC)/test/mocha.html \
- $(TARGET_DIST)/test
+.PHONY: watch
+watch: TARGET ?= firefox
+watch:
+ $(cli) watch
.PHONY: clean
clean:
- @rm -rf $(BUILD_DIR)
\ No newline at end of file
+ $(cli) clean
\ No newline at end of file
diff --git a/package.json b/package.json
index a999ad2..2fe137b 100644
--- a/package.json
+++ b/package.json
@@ -5,6 +5,7 @@
"@rollup/plugin-commonjs": "^23.0.2",
"@rollup/plugin-node-resolve": "^15.0.1",
"@rollup/plugin-replace": "^5.0.1",
+ "@rollup/plugin-terser": "^0.3.0",
"@rollup/plugin-typescript": "^9.0.2",
"@tsconfig/svelte": "^3.0.0",
"@types/chai": "^4.3.4",
@@ -13,14 +14,16 @@
"@types/node": "^18.11.9",
"@types/uuid": "^8.3.4",
"@types/webextension-polyfill": "^0.9.1",
+ "@types/ws": "^8.5.4",
"chai": "^4.3.7",
+ "commander": "^9.5.0",
+ "fs-extra": "^11.1.0",
"lodash": "^4.17.21",
"lodash-core": "^4.17.19",
"lodash-es": "^4.17.21",
"mocha": "^10.1.0",
"rollup": "^3.4.0",
"rollup-plugin-svelte": "^7.1.0",
- "rollup-plugin-terser": "^7.0.2",
"simpledotcss": "^2.1.1",
"svelte": "^3.53.1",
"svelte-preprocess": "^4.10.7",
@@ -30,7 +33,8 @@
"uuid": "^9.0.0",
"web-ext": "7.4.0",
"webextension-polyfill": "^0.10.0",
- "ws": "^8.11.0"
+ "ws": "^8.11.0",
+ "zip-dir": "^2.0.0"
},
"license": "MIT"
}
\ No newline at end of file
diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml
index b9f8755..d24aa67 100644
--- a/pnpm-lock.yaml
+++ b/pnpm-lock.yaml
@@ -4,6 +4,7 @@ specifiers:
'@rollup/plugin-commonjs': ^23.0.2
'@rollup/plugin-node-resolve': ^15.0.1
'@rollup/plugin-replace': ^5.0.1
+ '@rollup/plugin-terser': ^0.3.0
'@rollup/plugin-typescript': ^9.0.2
'@tsconfig/svelte': ^3.0.0
'@types/chai': ^4.3.4
@@ -12,14 +13,16 @@ specifiers:
'@types/node': ^18.11.9
'@types/uuid': ^8.3.4
'@types/webextension-polyfill': ^0.9.1
+ '@types/ws': ^8.5.4
chai: ^4.3.7
+ commander: ^9.5.0
+ fs-extra: ^11.1.0
lodash: ^4.17.21
lodash-core: ^4.17.19
lodash-es: ^4.17.21
mocha: ^10.1.0
rollup: ^3.4.0
rollup-plugin-svelte: ^7.1.0
- rollup-plugin-terser: ^7.0.2
simpledotcss: ^2.1.1
svelte: ^3.53.1
svelte-preprocess: ^4.10.7
@@ -30,11 +33,13 @@ specifiers:
web-ext: 7.4.0
webextension-polyfill: ^0.10.0
ws: ^8.11.0
+ zip-dir: ^2.0.0
devDependencies:
'@rollup/plugin-commonjs': 23.0.2_rollup@3.4.0
'@rollup/plugin-node-resolve': 15.0.1_rollup@3.4.0
'@rollup/plugin-replace': 5.0.1_rollup@3.4.0
+ '@rollup/plugin-terser': 0.3.0_rollup@3.4.0
'@rollup/plugin-typescript': 9.0.2_vmrh7niimjcuan47h4hgsgyxjy
'@tsconfig/svelte': 3.0.0
'@types/chai': 4.3.4
@@ -43,14 +48,16 @@ devDependencies:
'@types/node': 18.11.9
'@types/uuid': 8.3.4
'@types/webextension-polyfill': 0.9.1
+ '@types/ws': 8.5.4
chai: 4.3.7
+ commander: 9.5.0
+ fs-extra: 11.1.0
lodash: 4.17.21
lodash-core: 4.17.19
lodash-es: 4.17.21
mocha: 10.1.0
rollup: 3.4.0
rollup-plugin-svelte: 7.1.0_rollup@3.4.0+svelte@3.53.1
- rollup-plugin-terser: 7.0.2_rollup@3.4.0
simpledotcss: 2.1.1
svelte: 3.53.1
svelte-preprocess: 4.10.7_7dvewpees4iyn2tkw2qzal77a4
@@ -61,6 +68,7 @@ devDependencies:
web-ext: 7.4.0
webextension-polyfill: 0.10.0
ws: 8.11.0
+ zip-dir: 2.0.0
packages:
@@ -110,7 +118,7 @@ packages:
'@devicefarmer/adbkit-logcat': 2.1.2
'@devicefarmer/adbkit-monkey': 1.2.1
bluebird: 3.7.2
- commander: 9.4.1
+ commander: 9.5.0
debug: 4.3.4
node-forge: 1.3.1
split: 1.0.1
@@ -282,6 +290,21 @@ packages:
rollup: 3.4.0
dev: true
+ /@rollup/plugin-terser/0.3.0_rollup@3.4.0:
+ resolution: {integrity: sha512-mYTkNW9KjOscS/3QWU5LfOKsR3/fAAVDaqcAe2TZ7ng6pN46f+C7FOZbITuIW/neA+PhcjoKl7yMyB3XcmA4gw==}
+ engines: {node: '>=14.0.0'}
+ peerDependencies:
+ rollup: ^2.x || ^3.x
+ peerDependenciesMeta:
+ rollup:
+ optional: true
+ dependencies:
+ rollup: 3.4.0
+ serialize-javascript: 6.0.0
+ smob: 0.0.6
+ terser: 5.16.1
+ dev: true
+
/@rollup/plugin-typescript/9.0.2_vmrh7niimjcuan47h4hgsgyxjy:
resolution: {integrity: sha512-/sS93vmHUMjzDUsl5scNQr1mUlNE1QjBBvOhmRwJCH8k2RRhDIm3c977B3wdu3t3Ap17W6dDeXP3hj1P1Un1bA==}
engines: {node: '>=14.0.0'}
@@ -389,6 +412,12 @@ packages:
resolution: {integrity: sha512-6aNzPIhqKlAV9t06nwSH3/veAceYE2dS2RVFZI8V1+UXHqsFNB6cRwxNmheiBvEGRc45E/gyZNzH0xAYIC27KA==}
dev: true
+ /@types/ws/8.5.4:
+ resolution: {integrity: sha512-zdQDHKUgcX/zBc4GrwsE/7dVdAD8JR4EuiAXiiUhhfyIJXXb2+PrGshFyeXWQPMmmZ2XxgaqclgpIC7eTXc1mg==}
+ dependencies:
+ '@types/node': 18.11.9
+ dev: true
+
/@types/yauzl/2.10.0:
resolution: {integrity: sha512-Cn6WYCm0tXv8p6k+A8PvbDG763EDpBoTzHdA+Q/MF6H3sapGjCm9NzoaJncJS9tUKSuCoDs9XHxYYsQDgxR6kw==}
dependencies:
@@ -937,8 +966,8 @@ packages:
graceful-readlink: 1.0.1
dev: true
- /commander/9.4.1:
- resolution: {integrity: sha512-5EEkTNyHNGFPD2H+c/dXXfQZYa/scCKasxWcXJaWnNJ99pnQN9Vnmqow+p+PlFPE63Q6mThaZws1T+HxfpgtPw==}
+ /commander/9.5.0:
+ resolution: {integrity: sha512-KRs7WVDKg86PWiuAqhDrAQnTXZKraVcCc6vFdL14qrZ/DcWwuRo7VoiYXalXO7S5GKpqYiVEwCbgFDfxNHKJBQ==}
engines: {node: ^12.20.0 || >=14}
dev: true
@@ -1586,6 +1615,15 @@ packages:
universalify: 2.0.0
dev: true
+ /fs-extra/11.1.0:
+ resolution: {integrity: sha512-0rcTq621PD5jM/e0a3EJoGC/1TC5ZBCERW82LQuwfGnCa1V8w7dpYH1yNu+SLb6E5dkeCBzKEyLGlFrnr+dUyw==}
+ engines: {node: '>=14.14'}
+ dependencies:
+ graceful-fs: 4.2.10
+ jsonfile: 6.1.0
+ universalify: 2.0.0
+ dev: true
+
/fs-extra/9.0.1:
resolution: {integrity: sha512-h2iAoN838FqAFJY2/qVpzFXy+EBxfVE220PalAqQLDVsFOHLJrZvut5puAbCdNv6WJk+B8ihI+k0c7JK5erwqQ==}
engines: {node: '>=10'}
@@ -2063,15 +2101,6 @@ packages:
resolution: {integrity: sha1-elSbvZ/+FYWwzQoZHiAwVb7ldLQ=}
dev: true
- /jest-worker/26.6.2:
- resolution: {integrity: sha512-KWYVV1c4i+jbMpaBC+U++4Va0cp8OisU185o73T1vo99hqi7w8tSJfUXYswwqqrjzwxa6KpRK54WhPvwf5w6PQ==}
- engines: {node: '>= 10.13.0'}
- dependencies:
- '@types/node': 18.11.9
- merge-stream: 2.0.0
- supports-color: 7.2.0
- dev: true
-
/jose/4.11.1:
resolution: {integrity: sha512-YRv4Tk/Wlug8qicwqFNFVEZSdbROCHRAC6qu/i0dyNKr5JQdoa2pIGoS04lLO/jXQX7Z9omoNewYIVIxqZBd9Q==}
dev: true
@@ -3052,18 +3081,6 @@ packages:
svelte: 3.53.1
dev: true
- /rollup-plugin-terser/7.0.2_rollup@3.4.0:
- resolution: {integrity: sha512-w3iIaU4OxcF52UUXiZNsNeuXIMDvFrr+ZXK6bFZ0Q60qyVfq4uLptoS4bbq3paG3x216eQllFZX7zt6TIImguQ==}
- peerDependencies:
- rollup: ^2.0.0
- dependencies:
- '@babel/code-frame': 7.18.6
- jest-worker: 26.6.2
- rollup: 3.4.0
- serialize-javascript: 4.0.0
- terser: 5.16.0
- dev: true
-
/rollup-pluginutils/2.8.2:
resolution: {integrity: sha512-EEp9NhnUkwY8aif6bxgovPHMoMoNr2FulJziTndpt5H9RdwC47GSGuII9XxpSdzVGM0GWrNPHV6ie1LTNJPaLQ==}
dependencies:
@@ -3140,12 +3157,6 @@ packages:
lru-cache: 6.0.0
dev: true
- /serialize-javascript/4.0.0:
- resolution: {integrity: sha512-GaNA54380uFefWghODBWEGisLZFj00nS5ACs6yHa9nLqlLpVLO8ChDGeKRjZnV4Nh4n0Qi7nhYZD/9fCPzEqkw==}
- dependencies:
- randombytes: 2.1.0
- dev: true
-
/serialize-javascript/6.0.0:
resolution: {integrity: sha512-Qr3TosvguFt8ePWqsvRfrKyQXIiW+nGbYpy8XK24NQHE83caxWt+mIymTT19DGFbNWNLfEwsrkSmN64lVWB9ag==}
dependencies:
@@ -3207,6 +3218,10 @@ packages:
resolution: {integrity: sha512-BhqPaXVd8yGiZucZzn8rRV+1q19iVUvkv4mVXvtc4KMuf8H7/FrreTqzDcTfOl8X44DftHdn/laMkAKtX/itcw==}
dev: true
+ /smob/0.0.6:
+ resolution: {integrity: sha512-V21+XeNni+tTyiST1MHsa84AQhT1aFZipzPpOFAVB8DkHzwJyjjAmt9bgwnuZiZWnIbMo2duE29wybxv/7HWUw==}
+ dev: true
+
/sonic-boom/3.2.0:
resolution: {integrity: sha512-SbbZ+Kqj/XIunvIAgUZRlqd6CGQYq71tRRbXR92Za8J/R3Yh4Av+TWENiSiEgnlwckYLyP0YZQWVfyNC0dzLaA==}
dependencies:
@@ -3461,8 +3476,8 @@ packages:
engines: {node: '>= 8'}
dev: true
- /terser/5.16.0:
- resolution: {integrity: sha512-KjTV81QKStSfwbNiwlBXfcgMcOloyuRdb62/iLFPGBcVNF4EXjhdYBhYHmbJpiBrVxZhDvltE11j+LBQUxEEJg==}
+ /terser/5.16.1:
+ resolution: {integrity: sha512-xvQfyfA1ayT0qdK47zskQgRZeWLoOQ8JQ6mIgRGVNwZKdQMU+5FkCBjmv4QjcrTzyZquRw2FVtlJSRUmMKQslw==}
engines: {node: '>=10'}
hasBin: true
dependencies:
diff --git a/scripts/build_rollup.mjs b/scripts/build_rollup.mjs
new file mode 100644
index 0000000..361c2b4
--- /dev/null
+++ b/scripts/build_rollup.mjs
@@ -0,0 +1,82 @@
+import { readStdout } from "./utils.mjs";
+
+import commonjs from '@rollup/plugin-commonjs';
+import resolve from '@rollup/plugin-node-resolve';
+import replace from '@rollup/plugin-replace';
+import terser from '@rollup/plugin-terser';
+import typescript from '@rollup/plugin-typescript';
+import os from "node:os";
+import pathLib from 'node:path';
+import svelte from 'rollup-plugin-svelte';
+import autoPreprocess from 'svelte-preprocess';
+
+
+const useCustomElement = ["components"]
+
+export default function (opts) {
+ const { profile, websocketServer, src, dist, entryPoints, target } = opts
+
+ const isProd = profile.toLowerCase() === 'prod';
+ const sourceMap = !isProd
+
+ const safeEnvVar = {
+ commitId: readStdout("git rev-parse --short HEAD"),
+ date: readStdout("date --rfc-3339=seconds"),
+ nodeVersion: readStdout("node --version"),
+ rollupVersion: readStdout("npx rollup --version"),
+ os: os.platform(),
+ profile,
+ websocketServer,
+ target,
+ }
+
+ function getPlugins(entrypoint) {
+ return [
+ commonjs(),
+ replace({
+ values: {
+ // avoid "Function constructor" warning
+ "Function('return this')()": "globalThis",
+ },
+ delimiters: ["", ""],
+ preventAssignment: true
+ }),
+ replace({
+ __ENV: JSON.stringify(safeEnvVar),
+ __BUILD_PROFILE: JSON.stringify(profile),
+ preventAssignment: true
+ }),
+ svelte({
+ preprocess: autoPreprocess(),
+ compilerOptions: {
+ customElement: useCustomElement.includes(entrypoint)
+ }
+ }),
+ typescript({ sourceMap: false }),
+ resolve({ browser: true }),
+ isProd && terser(),
+ ]
+ }
+
+ const output = []
+
+ for (const e of entryPoints) {
+ output.push({
+ external: ["chai", "mocha", "Mocha"],
+ input: pathLib.join(src, e, "main.ts"),
+ output: {
+ sourcemap: sourceMap,
+ globals: {
+ "chai": "chai",
+ "mocha": "mocha",
+ "Mocha": "Mocha"
+ },
+ file: pathLib.join(dist, e, "main.js"),
+ format: 'iife'
+ },
+ plugins: getPlugins(e)
+ })
+ }
+
+ return output
+}
\ No newline at end of file
diff --git a/scripts/cli.mjs b/scripts/cli.mjs
new file mode 100644
index 0000000..a486844
--- /dev/null
+++ b/scripts/cli.mjs
@@ -0,0 +1,295 @@
+import { program } from 'commander'
+import fs from 'fs-extra'
+import { spawn } from 'node:child_process'
+import pathLib from 'node:path'
+import * as rollup from 'rollup'
+import webExt from 'web-ext'
+import { WebSocketServer } from 'ws'
+import zipDir from 'zip-dir'
+import buildRollupConfig from './build_rollup.mjs'
+import genManifest from './gen_manifest.mjs'
+import { isTestTarget, mustEnv } from './utils.mjs'
+
+const BUILD_DIR = mustEnv("BUILD_DIR", "./build")
+const BUILD_SRC = mustEnv("BUILD_SRC", "./src")
+const BUILD_VERSION = mustEnv("BUILD_VERSION", "2.1.0")
+
+const FIREFOX = "firefox-developer-edition"
+function copyAssets(destDir) {
+ const assets = [
+ {
+ src: pathLib.join(BUILD_SRC, "options/options.html"),
+ dest: pathLib.join(destDir, "options/options.html")
+ },
+ {
+ src: pathLib.join(BUILD_SRC, "/icon"),
+ dest: pathLib.join(destDir, "/icon")
+ },
+ {
+ src: pathLib.join(BUILD_SRC, "/_locales"),
+ dest: pathLib.join(destDir, "/_locales")
+ },
+
+ {
+ src: "node_modules/simpledotcss/simple.min.css",
+ dest: pathLib.join(destDir, "res/simple.min.css")
+ },
+ ]
+ for (const entry of assets) {
+ fs.copySync(entry.src, entry.dest)
+ }
+}
+
+function copyTestAssets(dest) {
+ dest = pathLib.join(dest, "/test")
+ const assets = [
+ {
+ src: pathLib.join("node_modules/mocha/mocha.js"),
+ dest: pathLib.join(dest, "mocha.js")
+ },
+ {
+ src: pathLib.join("node_modules/mocha/mocha.css"),
+ dest: pathLib.join(dest, "mocha.css")
+ },
+ {
+ src: pathLib.join("node_modules/chai/chai.js"),
+ dest: pathLib.join(dest, "chai.js")
+ },
+ {
+ src: pathLib.join(BUILD_SRC, "/test/mocha.html"),
+ dest: pathLib.join(dest, "mocha.html")
+ }
+ ]
+ for (const entry of assets) {
+ fs.copySync(entry.src, entry.dest)
+ }
+}
+
+function buildDist(target) {
+ return pathLib.join(BUILD_DIR, target, "dist")
+}
+
+function buildManifest(dist, version, target) {
+ const manifest = genManifest({
+ version: version,
+ target: target,
+ })
+ fs.writeFileSync(pathLib.join(dist, "manifest.json"), JSON.stringify(manifest, null, " "))
+}
+
+async function buildWithRollup(config) {
+ for (const optionsObj of config) {
+ try {
+ const bundle = await rollup.rollup(optionsObj);
+ await bundle.write(optionsObj.output);
+ if (bundle) {
+ // closes the bundle
+ await bundle.close();
+ }
+ } catch (err) {
+ console.error(err)
+ }
+ }
+}
+
+function validateTarget(target) {
+ if (!["firefox", "chromium", "firefox-test"].includes(target)) {
+ throw new Error("not support build target: " + target)
+ }
+}
+
+const defaultEntryPoints = ["background", "content_scripts", "options", "components"]
+
+program.command('build')
+ .description('Build extension')
+ .option('-t, --target ', "The browser target to build", "firefox")
+ .option('--profile ', "The profile of compilation", "debug")
+ .option('--artifacts ', "The directory to store artifacts", "artifacts")
+ .option('--lint', "Use web-ext the validate extension source", false)
+ .option('--watch', "Watch source file change", false)
+ .option('--websocket-server ', "The address of websocket server for capture event of unit test", "ws://localhost:8000")
+ .action(async (args, options) => {
+ validateTarget(args.target)
+ const dist = pathLib.join(BUILD_DIR, args.target, "dist")
+ copyAssets(dist)
+ buildManifest(dist, BUILD_VERSION, args.target)
+
+ const entryPoints = [...defaultEntryPoints]
+ if (isTestTarget(args.target)) {
+ entryPoints.push("test")
+ copyTestAssets()
+ }
+ console.debug("args:", args)
+ console.debug("entryPoints: ", entryPoints)
+
+ const config = buildRollupConfig({
+ profile: args.profile,
+ src: BUILD_SRC,
+ dist: dist,
+ entryPoints: entryPoints,
+ watch: args.watch,
+ target: args.target,
+ websocketServer: args.websocketServer,
+ })
+ const artifacts = pathLib.join(
+ BUILD_DIR,
+ args.artifacts
+ )
+ fs.ensureDirSync(artifacts)
+
+ if (args.watch) {
+ const watcher = rollup.watch(config)
+ watcher.on('event', (event) => {
+ if (event.code === "BUNDLE_END") {
+ console.log("code: ", event.code, "input:", event.input, "output:", event.output)
+ }
+ const { result } = event;
+ if (result) {
+ result.close();
+ }
+ });
+ } else {
+ await buildWithRollup(config)
+
+ if (args.lint) {
+ webExt.cmd.lint({ sourceDir: dist })
+ }
+ await zipDir(
+ dist,
+ {
+ saveTo: pathLib.join(
+ artifacts,
+ `glitterdrag-pro-${BUILD_VERSION}-${args.target}.zip`
+ )
+ }
+ );
+ }
+
+ });
+
+program.command('test')
+ .description('Start Firefox browser and run unit tests')
+ .option('-t, --target ', "The target to build", "firefox-test")
+ .option('-s, --websocket-server', "The address of websocket server for capture event of unit test", "ws://localhost:8000")
+ .action(async (args) => {
+ validateTarget(args.target)
+ const dist = buildDist(args.target)
+
+ async function runBrowser(signal) {
+
+ return new Promise((resolve, reject) => {
+
+ const proc = spawn("pnpm", ["exec", "web-ext", "run", "-f", FIREFOX, "-s", dist], {
+ stdio: ["ignore", process.stdout, process.stderr],
+ signal: signal
+ })
+
+ proc.on("error", (err) => {
+ console.error(err)
+ resolve()
+ })
+
+ proc.on("close", (code) => {
+ if (code === 0) {
+ resolve()
+ } else {
+ reject("code: " + code)
+ }
+ })
+ })
+ }
+
+
+ async function waitTestComplete() {
+ const url = new URL(args.websocketServer)
+ let port = undefined
+ if (url.port.length > 0) {
+ port = Number.parseInt(url.port)
+ }
+ const wss = new WebSocketServer({ host: url.hostname, port, });
+
+ return new Promise((resolve) => {
+ wss.on('connection', (ws) => {
+ ws.on('message', (data) => {
+ const [type, payload] = JSON.parse(data)
+ switch (type) {
+ case "start": {
+ ws.send("ok")
+ break
+ }
+ case "pass": {
+ ws.send("ok")
+ console.info("pass: %s", payload.fullTitle)
+ break
+ }
+ case "fail": {
+ ws.send("ok")
+ console.error("fail: %s", payload.fullTitle)
+ break
+ }
+ case "end": {
+ ws.close()
+ wss.close()
+ resolve(payload)
+ break
+ }
+ default: {
+ throw new Error("unknown event type: ", type)
+ }
+ }
+ });
+ ws.send("ok")
+ });
+ })
+ }
+
+ const controller = new AbortController()
+ const resultPromise = waitTestComplete()
+ const browser = runBrowser(controller.signal)
+
+ const result = await resultPromise
+ await browser
+ console.log("tests: %d, passes: %d, failures: %d", result.tests, result.passes, result.failures)
+ if (result.failure > 0) {
+ process.exit(1)
+ } else {
+ process.exit(0)
+ }
+ });
+
+program.command('watch')
+ .description('detect dist directory change and reload extension')
+ .option('-t, --target', "The target to build", "firefox")
+ .action((args) => {
+
+ validateTarget(args.target)
+
+ const dist = buildDist(args.target)
+
+ return new Promise((resolve, reject) => {
+ const proc = spawn("pnpm", ["exec", "web-ext", "run", "-f", FIREFOX, "-s", dist], {
+ stdio: ["ignore", process.stdout, process.stderr],
+ })
+
+ proc.on("error", (err) => {
+ console.error(err)
+ resolve()
+ })
+
+ proc.on("close", (code) => {
+ if (code === 0) {
+ resolve()
+ } else {
+ reject("code: " + code)
+ }
+ })
+ })
+ });
+
+program.command('clean')
+ .description('clean build directory')
+ .action(() => {
+ fs.removeSync(BUILD_DIR)
+ });
+
+await program.parseAsync();
\ No newline at end of file
diff --git a/scripts/gen_manifest.js b/scripts/gen_manifest.js
deleted file mode 100644
index 4b74b7b..0000000
--- a/scripts/gen_manifest.js
+++ /dev/null
@@ -1,119 +0,0 @@
-import { mustEnv } from "./utils.js"
-
-const version = mustEnv("BUILD_VERSION")
-const targetBrowser = mustEnv("TARGET_BROWSER")
-const isFirefox = targetBrowser.startsWith("firefox")
-const isChromium = targetBrowser.startsWith("chromium")
-const isTest = targetBrowser.endsWith("test")
-
-function contentSecurityPolicy() {
- if (isFirefox && isTest) {
- // override CSP to allow mocha test connect to a insecure websocket
- // https://bugzilla.mozilla.org/show_bug.cgi?id=1797086
- return {
- "extension_pages": "script-src 'self'"
- }
- }
- return undefined
-}
-
-function manifestPermissions() {
- const permissions = [
- "activeTab",
- "storage",
- "tabs",
- "clipboardWrite",
- "downloads",
- "search",
- "scripting",
- "contextMenus",
- "alarms"
- ]
-
- if (isFirefox) {
- permissions.push(...[
- "contextualIdentities",
- "cookies"
- ])
- }
-
- return permissions
-}
-
-function manifestBackground() {
- if (isChromium) {
- return {
- "service_worker": "background/main.js"
- }
- }
- return {
- "scripts": [
- "background/main.js"
- ]
- }
-}
-
-function manifestVersion() {
- return version
-}
-
-function browserSpecificSetting() {
- if (isFirefox) {
- return {
- "gecko": {
- "id": "glitterdragpro@harytfw",
- "strict_min_version": "106.0"
- }
- }
- }
- return undefined
-}
-
-const manifest = {
- "manifest_version": 3,
- "name": "__MSG_extensionName__",
- "description": "__MSG_extensionDescription__",
- "version": manifestVersion(),
- "homepage_url": "https://github.com/harytfw/GlitterDrag",
- "icons": {
- "128": "/icon/drag.png"
- },
- "author": "harytfw",
- "permissions": manifestPermissions(),
- "host_permissions": [
- "*://*/*"
- ],
- "background": manifestBackground(),
- "content_scripts": [
- {
- "run_at": "document_end",
- "all_frames": true,
- "matches": [
- "*://*/*"
- ],
- "js": [
- "content_scripts/main.js"
- ]
- }
- ],
- "options_ui": {
- "page": "options/options.html",
- "browser_style": true,
- "open_in_tab": true
- },
- "web_accessible_resources": [
- {
- "resources": [
- "components/main.js"
- ],
- "matches": [
- "*://*/*"
- ]
- }
- ],
- "default_locale": "en",
- "browser_specific_settings": browserSpecificSetting(),
- "content_security_policy": contentSecurityPolicy()
-}
-
-console.log(JSON.stringify(manifest, null, " "))
\ No newline at end of file
diff --git a/scripts/gen_manifest.mjs b/scripts/gen_manifest.mjs
new file mode 100644
index 0000000..a9c9a56
--- /dev/null
+++ b/scripts/gen_manifest.mjs
@@ -0,0 +1,124 @@
+import assert from 'node:assert/strict'
+export default function (opt) {
+
+ const { version, target } = opt
+
+ {
+ assert.ok(typeof version === 'string')
+ assert.ok(version.length > 0)
+
+ assert.ok(typeof target === 'string')
+ assert.ok(target.length > 0)
+ }
+
+ const isFirefox = target.startsWith("firefox")
+ const isChromium = target.startsWith("chromium")
+ const isTest = target.endsWith("test")
+
+ function contentSecurityPolicy(opt) {
+ if (isFirefox && isTest) {
+ // override CSP to allow mocha test connect to a insecure websocket
+ // https://bugzilla.mozilla.org/show_bug.cgi?id=1797086
+ return {
+ "extension_pages": "script-src 'self'"
+ }
+ }
+ return undefined
+ }
+
+ function manifestPermissions() {
+ const permissions = [
+ "activeTab",
+ "storage",
+ "tabs",
+ "clipboardWrite",
+ "downloads",
+ "search",
+ "scripting",
+ "contextMenus",
+ "alarms"
+ ]
+
+ if (isFirefox) {
+ permissions.push(...[
+ "contextualIdentities",
+ "cookies"
+ ])
+ }
+
+ return permissions
+ }
+
+ function manifestBackground() {
+ if (isChromium) {
+ return {
+ "service_worker": "background/main.js"
+ }
+ }
+ return {
+ "scripts": [
+ "background/main.js"
+ ]
+ }
+ }
+
+ function browserSpecificSetting() {
+ if (isFirefox) {
+ return {
+ "gecko": {
+ "id": "glitterdragpro@harytfw",
+ "strict_min_version": "106.0"
+ }
+ }
+ }
+ return undefined
+ }
+
+ const manifest = {
+ "manifest_version": 3,
+ "name": "__MSG_extensionName__",
+ "description": "__MSG_extensionDescription__",
+ "version": version,
+ "homepage_url": "https://github.com/harytfw/GlitterDrag",
+ "icons": {
+ "128": "/icon/drag.png"
+ },
+ "author": "harytfw",
+ "permissions": manifestPermissions(),
+ "host_permissions": [
+ "*://*/*"
+ ],
+ "background": manifestBackground(),
+ "content_scripts": [
+ {
+ "run_at": "document_end",
+ "all_frames": true,
+ "matches": [
+ "*://*/*"
+ ],
+ "js": [
+ "content_scripts/main.js"
+ ]
+ }
+ ],
+ "options_ui": {
+ "page": "options/options.html",
+ "browser_style": true,
+ "open_in_tab": true
+ },
+ "web_accessible_resources": [
+ {
+ "resources": [
+ "components/main.js"
+ ],
+ "matches": [
+ "*://*/*"
+ ]
+ }
+ ],
+ "default_locale": "en",
+ "browser_specific_settings": browserSpecificSetting(),
+ "content_security_policy": contentSecurityPolicy()
+ }
+ return manifest
+}
diff --git a/scripts/rollup.config.js b/scripts/rollup.config.js
deleted file mode 100644
index 560c9b8..0000000
--- a/scripts/rollup.config.js
+++ /dev/null
@@ -1,94 +0,0 @@
-import { mustEnv, readStdout } from "./utils.js"
-
-import svelte from 'rollup-plugin-svelte';
-import resolve from '@rollup/plugin-node-resolve';
-import typescript from '@rollup/plugin-typescript';
-import replace from '@rollup/plugin-replace';
-import { terser } from 'rollup-plugin-terser';
-import autoPreprocess from 'svelte-preprocess';
-import commonjs from '@rollup/plugin-commonjs';
-import pathLib from 'node:path'
-import os from "node:os"
-
-const BUILD_COMMIT_ID = readStdout("git rev-parse --short HEAD")
-const BUILD_DATE = readStdout("date --rfc-3339=seconds")
-const BUILD_NODE_VERSION = readStdout("node --version")
-const BUILD_ROLLUP_VERSION = readStdout("npx rollup --version")
-const BUILD_OS = os.platform()
-const BUILD_PROFILE = mustEnv("BUILD_PROFILE")
-const BUILD_WEBSOCKET_SERVER = mustEnv("BUILD_WEBSOCKET_SERVER", "")
-
-const isProd = BUILD_PROFILE.toLowerCase() === 'prod';
-const sourceMap = !isProd
-const src = mustEnv("SRC")
-const dist = mustEnv("TARGET_DIST")
-const entryPoints = mustEnv("ENTRY_POINTS").split(" ")
-
-const safeEnvVar = {
- BUILD_COMMIT_ID,
- BUILD_DATE,
- BUILD_NODE_VERSION,
- BUILD_ROLLUP_VERSION,
- BUILD_OS,
- BUILD_PROFILE,
- BUILD_WEBSOCKET_SERVER,
-}
-
-const useCustomElement = ["components"]
-
-function getPlugins(entrypoint) {
- const plugins = [
- commonjs(),
- replace({
- values: {
- // avoid "Function constructor" warning
- "Function('return this')()": "globalThis",
- },
- delimiters: ["", ""],
- preventAssignment: true
- }),
- replace({
- __ENV: JSON.stringify(safeEnvVar),
- __BUILD_PROFILE: JSON.stringify(BUILD_PROFILE),
- preventAssignment: true
- }),
- svelte({
- preprocess: autoPreprocess(),
- compilerOptions: {
- customElement: useCustomElement.includes(entrypoint)
- }
- }),
- typescript({ sourceMap: false }),
- resolve({ browser: true }),
- ]
-
- if (isProd) {
- plugins.push(terser())
- }
-
- return plugins
-}
-
-console.log("sourcemap: ", sourceMap)
-
-const output = []
-
-for (const e of entryPoints) {
- output.push({
- external: ["chai", "mocha", "Mocha"],
- input: pathLib.join(src, e, "main.ts"),
- output: {
- sourcemap: sourceMap,
- globals: {
- "chai": "chai",
- "mocha": "mocha",
- "Mocha": "Mocha"
- },
- file: pathLib.join(dist, e, "main.js"),
- format: 'iife'
- },
- plugins: getPlugins(e)
- })
-}
-
-export default output
diff --git a/scripts/test_bootstrap.js b/scripts/test_bootstrap.js
deleted file mode 100644
index ffd3c86..0000000
--- a/scripts/test_bootstrap.js
+++ /dev/null
@@ -1,95 +0,0 @@
-import { WebSocketServer } from 'ws';
-import { spawn } from 'node:child_process';
-import webExt from 'web-ext'
-
-
-const TEST_TIMEOUT_MS = 1000 * 60 * 5
-
-const { TARGET_DIST, BUILD_WEBSOCKET_SERVER } = process.env
-
-async function runBrowser(signal) {
-
- return new Promise((resolve, reject) => {
-
- const proc = spawn("pnpm", ["exec", "web-ext", "run", "-f", "firefox-developer-edition" , "-s", TARGET_DIST], {
- stdio: ["ignore", process.stdout, process.stderr],
- signal: signal
- })
-
- proc.on("error", (err) => {
- console.error(err)
- resolve()
- })
-
- proc.on("close", (code) => {
- if (code === 0) {
- resolve()
- } else {
- reject("code: " + code)
- }
- })
- })
-}
-
-async function waitTestComplete() {
- const url = new URL(BUILD_WEBSOCKET_SERVER)
- let port = undefined
- if (url.port.length > 0) {
- port = Number.parseInt(url.port)
- }
- const wss = new WebSocketServer({ host: url.hostname, port, });
-
- return new Promise((resolve) => {
- wss.on('connection', (ws) => {
- ws.on('message', (data) => {
- const [type, payload] = JSON.parse(data)
- switch (type) {
- case "start": {
- ws.send("ok")
- break
- }
- case "pass": {
- ws.send("ok")
- console.info("pass: %s", payload.fullTitle)
- break
- }
- case "fail": {
- ws.send("ok")
- console.error("fail: %s", payload.fullTitle)
- break
- }
- case "end": {
- ws.close()
- wss.close()
- resolve(payload)
- break
- }
- default: {
- throw new Error("unknown event type: ", type)
- }
- }
- });
- ws.send("ok")
- });
- })
-}
-
-async function main() {
- const controller = new AbortController()
-
- const browser = runBrowser(controller.signal)
-
- const summary = await waitTestComplete()
-
- await browser
-
- console.log("tests: %d, passes: %d, failures: %d", summary.tests, summary.passes, summary.failures)
- // console.log(summary)
- if (summary.failure > 0) {
- process.exit(1)
- } else {
- process.exit(0)
- }
-}
-
-main()
\ No newline at end of file
diff --git a/scripts/utils.js b/scripts/utils.mjs
similarity index 85%
rename from scripts/utils.js
rename to scripts/utils.mjs
index ec7cab8..caa21af 100644
--- a/scripts/utils.js
+++ b/scripts/utils.mjs
@@ -18,3 +18,7 @@ export function readStdout(command) {
const result = execSync(command)
return result.toString("utf8")
}
+
+export function isTestTarget(target) {
+ return target.endsWith("test")
+}
diff --git a/src/build_info.ts b/src/build_info.ts
index 368d511..fb0e4f1 100644
--- a/src/build_info.ts
+++ b/src/build_info.ts
@@ -1,12 +1,4 @@
const env = __ENV
-export default {
- commitId: env["BUILD_COMMIT_ID"],
- date: env["BUILD_DATE"],
- nodeVersion: env["BUILD_NODE_VERSION"],
- rollupVersion: env["BUILD_ROLLUP_VERSION"],
- os: env["BUILD_OS"],
- websocketServer: env["BUILD_WEBSOCKET_SERVER"],
- profile: env["BUILD_PROFILE"],
-} as const
+export default env
diff --git a/src/global.d.ts b/src/global.d.ts
index 2f1a245..676b83f 100644
--- a/src/global.d.ts
+++ b/src/global.d.ts
@@ -1,8 +1,18 @@
import { Mocha } from 'mocha'
+type EnvVariable = {
+ commitId: string
+ date: string
+ nodeVersion: string
+ rollupVersion: string
+ os: string
+ websocketServer: string
+ profile: string
+ target: string
+}
export declare global {
- export const __ENV: Record
+ export const __ENV: EnvVariable
export const __BUILD_PROFILE: 'prod' | 'debug'
export const chrome: any
export const Mocha: Mocha
diff --git a/src/test/mocha_init.ts b/src/test/mocha_init.ts
index 45988f9..f9511ff 100644
--- a/src/test/mocha_init.ts
+++ b/src/test/mocha_init.ts
@@ -67,7 +67,7 @@ async function closeInstance() {
}
export let ws = null
-if (buildInfo.websocketServer) {
+if (buildInfo.target.endsWith("test") && buildInfo.websocketServer) {
ws = new WebSocket(buildInfo.websocketServer)
ws.addEventListener("open", (event) => {
console.log("open", event)
From 1165bdac9b58365bdc848f3f1ad5210c4cff35dd Mon Sep 17 00:00:00 2001
From: harytfw
Date: Tue, 10 Jan 2023 14:31:27 +0800
Subject: [PATCH 10/21] refactor: clean code about menu
Signed-off-by: harytfw
---
src/components/menu/menu.svelte | 29 +++++-------
src/components/menu/menu.ts | 31 +++++++------
src/components/menu/menu_builder.test.ts | 5 ++-
src/components/menu/menu_builder.ts | 56 +++++++++++++-----------
src/components/types.ts | 3 ++
5 files changed, 66 insertions(+), 58 deletions(-)
diff --git a/src/components/menu/menu.svelte b/src/components/menu/menu.svelte
index 9f75599..eb44c77 100644
--- a/src/components/menu/menu.svelte
+++ b/src/components/menu/menu.svelte
@@ -3,11 +3,7 @@
@@ -99,14 +95,13 @@
id="container"
on:dragover={dragover}
on:dragleave={dragleave}
- width={box()[0]}
- height={box()[1]}
+ style="width: {box[0]}px; height: {box[1]}px;"
bind:this={container}
>
diff --git a/src/components/menu/menu.ts b/src/components/menu/menu.ts
index cb3d3d6..cc2b727 100644
--- a/src/components/menu/menu.ts
+++ b/src/components/menu/menu.ts
@@ -8,35 +8,27 @@ import { LogLevel } from '../../config/config';
const log = rootLog.subLogger(LogLevel.VVV, "menu")
const tag = "glitterdrag-menu"
-function computeStyle(x: number, y: number, width: number, height: number) {
- const style = `position: absolute;
- left: ${x}px;
- top: ${y}px;
- z-index: 2147483647;
- width: ${width}px;
- height: ${height}px;
- user-select: none;
- `
- return style
-}
class MenuImpl extends MessageTarget implements MenuInterface {
+
elem: MenuElement & HTMLElement
constructor() {
super(ProxyEventType.Menu)
customElements.define(tag, MenuElement as any);
this.elem = document.createElement(tag) as HTMLElement & MenuElement
- this.elem.setAttribute("style", computeStyle(0, 0, 0, 0));
+ this.elem.setAttribute("style", this.computeStyle(0, 0, 0, 0));
}
show(opts: ShowMenuOptions) {
log.V("show menu: ", opts)
- const [width, height] = this.elem.box()
+ const [width, height] = this.elem.box
+
const x = opts.position.x - width / 2
const y = opts.position.y - height / 2
- this.elem.setAttribute("style", computeStyle(x, y, width, height));
+
+ this.elem.setAttribute("style", this.computeStyle(x, y, width, height));
this.elem.show(opts)
!this.elem.parentElement && document.body.append(this.elem)
}
@@ -50,6 +42,17 @@ class MenuImpl extends MessageTarget implements MenuInterface {
this.elem.remove()
}
+ computeStyle(x: number, y: number, width: number, height: number) {
+ const style = `position: absolute;
+ left: ${x}px;
+ top: ${y}px;
+ z-index: 2147483647;
+ width: ${width}px;
+ height: ${height}px;
+ user-select: none;
+ `
+ return style
+ }
}
export const menuImpl = new MenuImpl()
\ No newline at end of file
diff --git a/src/components/menu/menu_builder.test.ts b/src/components/menu/menu_builder.test.ts
index 119c4d8..d86af46 100644
--- a/src/components/menu/menu_builder.test.ts
+++ b/src/components/menu/menu_builder.test.ts
@@ -56,7 +56,10 @@ describe("menu builder", () => {
dividerLineLength: 10,
iconOffset: 10,
iconSize: 10,
- textOffset: 10
+ textOffset: 10,
+ fontSize: 4,
+ width: 100,
+ height: 100
})
assert.equal(svg.querySelectorAll("g").length, 2)
})
diff --git a/src/components/menu/menu_builder.ts b/src/components/menu/menu_builder.ts
index aa27ed0..731cfa7 100644
--- a/src/components/menu/menu_builder.ts
+++ b/src/components/menu/menu_builder.ts
@@ -17,31 +17,6 @@ const colorDef = `
id="stop14618" />
`;
-const styleContent = `
- g .sector-fill {
- fill: #fff0;
- stroke: none;
- }
-
- g:hover .sector-fill {
- fill: url("#sector-color");
- }
-
- g.hover .sector-fill {
- fill: url("#sector-color");
- }
-
- g.hover .sector-fill text {
- fill: white;
- }
-
- g text {
- font-size: 4px;
- fill: white;
- text-shadow: #000 0px 0 4px;
- font-weight: bold;
- }
-`;
export function polar2cartesian(
r: number,
@@ -80,7 +55,7 @@ export async function toDataURL(text: string, type: string): Promise {
export async function rebuildMenu(svg: SVGSVGElement, opt: MenuOptions) {
- const [width, height] = [200, 200];
+ const { width, height } = opt
const center = [width / 2, height / 2];
const arcRadius = opt.dividerLineLength + opt.circleRadius;
const angleUnit = 360 / opt.items.length;
@@ -96,6 +71,35 @@ export async function rebuildMenu(svg: SVGSVGElement, opt: MenuOptions) {
svg.dataset["iconOffset"] = `${opt.iconOffset}`;
svg.dataset["iconSize"] = `${opt.iconSize}`;
svg.dataset["textOffset"] = `${opt.textOffset}`;
+ svg.dataset["fontSize"] = `${opt.fontSize}`;
+
+
+ const styleContent = `
+ g .sector-fill {
+ fill: #fff0;
+ stroke: none;
+ }
+
+ g:hover .sector-fill {
+ fill: url("#sector-color");
+ }
+
+ g.hover .sector-fill {
+ fill: url("#sector-color");
+ }
+
+ g.hover .sector-fill text {
+ fill: white;
+ }
+
+ g text {
+ font-size: ${opt.fontSize}px;
+ fill: white;
+ text-shadow: #000 0px 0 4px;
+ font-weight: bold;
+ }
+`;
+
{
const defs = document.createElementNS(SVG_NAMESPACE, "defs");
diff --git a/src/components/types.ts b/src/components/types.ts
index 92b574b..e78c96c 100644
--- a/src/components/types.ts
+++ b/src/components/types.ts
@@ -75,4 +75,7 @@ export interface MenuOptions {
iconOffset: number;
iconSize: number;
textOffset: number;
+ fontSize: number;
+ width: number;
+ height: number;
}
From 66aa94385deffd0e5354b2e87aedf55946c83ca5 Mon Sep 17 00:00:00 2001
From: harytfw
Date: Wed, 18 Jan 2023 18:25:58 +0800
Subject: [PATCH 11/21] refactor: clean code about cli.mjs
Signed-off-by: harytfw
---
scripts/cli.mjs | 23 ++++++++++-------------
1 file changed, 10 insertions(+), 13 deletions(-)
diff --git a/scripts/cli.mjs b/scripts/cli.mjs
index a486844..ddc05ed 100644
--- a/scripts/cli.mjs
+++ b/scripts/cli.mjs
@@ -15,6 +15,7 @@ const BUILD_SRC = mustEnv("BUILD_SRC", "./src")
const BUILD_VERSION = mustEnv("BUILD_VERSION", "2.1.0")
const FIREFOX = "firefox-developer-edition"
+
function copyAssets(destDir) {
const assets = [
{
@@ -79,23 +80,19 @@ function buildManifest(dist, version, target) {
async function buildWithRollup(config) {
for (const optionsObj of config) {
- try {
- const bundle = await rollup.rollup(optionsObj);
- await bundle.write(optionsObj.output);
- if (bundle) {
- // closes the bundle
- await bundle.close();
- }
- } catch (err) {
- console.error(err)
+ const bundle = await rollup.rollup(optionsObj);
+ await bundle.write(optionsObj.output);
+ if (bundle) {
+ await bundle.close();
}
}
}
function validateTarget(target) {
- if (!["firefox", "chromium", "firefox-test"].includes(target)) {
- throw new Error("not support build target: " + target)
+ if (["firefox", "chromium", "firefox-test"].includes(target)) {
+ return
}
+ throw new Error("not support build target: " + target)
}
const defaultEntryPoints = ["background", "content_scripts", "options", "components"]
@@ -105,7 +102,7 @@ program.command('build')
.option('-t, --target ', "The browser target to build", "firefox")
.option('--profile ', "The profile of compilation", "debug")
.option('--artifacts ', "The directory to store artifacts", "artifacts")
- .option('--lint', "Use web-ext the validate extension source", false)
+ .option('--lint', "Use web-ext validates extension source", false)
.option('--watch', "Watch source file change", false)
.option('--websocket-server ', "The address of websocket server for capture event of unit test", "ws://localhost:8000")
.action(async (args, options) => {
@@ -263,7 +260,7 @@ program.command('watch')
.action((args) => {
validateTarget(args.target)
-
+
const dist = buildDist(args.target)
return new Promise((resolve, reject) => {
From 26a48df37217fa0b42698e014aec65cd15d17ef8 Mon Sep 17 00:00:00 2001
From: harytfw
Date: Wed, 18 Jan 2023 18:32:32 +0800
Subject: [PATCH 12/21] test: clean code about unit test
Signed-off-by: harytfw
---
Makefile | 2 +-
src/background/executor.test.ts | 2 +-
src/background/search.test.ts | 2 +-
src/background/utils.test.ts | 16 ++++++++--------
src/config/config.test.ts | 9 +++------
src/content_scripts/utils.test.ts | 20 ++++++++++----------
src/utils/test.ts | 8 ++++++++
src/utils/test_helper.ts | 22 ----------------------
src/utils/var_substitute.test.ts | 1 -
9 files changed, 32 insertions(+), 50 deletions(-)
create mode 100644 src/utils/test.ts
delete mode 100644 src/utils/test_helper.ts
diff --git a/Makefile b/Makefile
index 40d6f81..3fb7c19 100644
--- a/Makefile
+++ b/Makefile
@@ -24,7 +24,7 @@ ext-chromium:
$(cli) build --target chromium --profile $(BUILD_PROFILE)
.PHONY: test
-test: TARGET ?= firefox
+test: TARGET ?= firefox-test
test:
$(cli) test --target $(TARGET)
diff --git a/src/background/executor.test.ts b/src/background/executor.test.ts
index e84b3b6..88f0d30 100644
--- a/src/background/executor.test.ts
+++ b/src/background/executor.test.ts
@@ -1,7 +1,7 @@
import browser, { tabs } from 'webextension-polyfill';
import { ActionConfig, CommandKind } from "../config/config"
-import { closeTab } from '../utils/test_helper';
+import { closeTab } from '../utils/test';
import { Executor } from "./executor"
import { blankExecuteContext } from "../context/test_helper"
diff --git a/src/background/search.test.ts b/src/background/search.test.ts
index ceb6ac8..bbf17ee 100644
--- a/src/background/search.test.ts
+++ b/src/background/search.test.ts
@@ -1,5 +1,5 @@
import browser from 'webextension-polyfill'
-import { closeTab } from '../utils/test_helper'
+import { closeTab } from '../utils/test'
import { searchText } from './search'
describe("test browser search", () => {
diff --git a/src/background/utils.test.ts b/src/background/utils.test.ts
index 47fe91a..d0d874f 100644
--- a/src/background/utils.test.ts
+++ b/src/background/utils.test.ts
@@ -1,5 +1,5 @@
import browser from 'webextension-polyfill'
-import { assertOk } from "../utils/test_helper"
+import { assert } from "chai"
import { blankExecuteContext } from "../context/test_helper"
import { bufferToObjectURL, buildDownloadableURL, generatedDownloadFileName } from "./utils"
@@ -9,44 +9,44 @@ describe("background utils", () => {
it("text", async () => {
const ctx = await blankExecuteContext()
ctx.data.selection = "hello world"
- assertOk(generatedDownloadFileName(ctx, await buildDownloadableURL(ctx)))
+ assert.ok(generatedDownloadFileName(ctx, await buildDownloadableURL(ctx)))
})
it("image", async () => {
const ctx = await blankExecuteContext()
ctx.data.imageSource = "http://example.com/a.jpg"
- assertOk(generatedDownloadFileName(ctx, await buildDownloadableURL(ctx)))
+ assert.ok(generatedDownloadFileName(ctx, await buildDownloadableURL(ctx)))
})
it("link", async () => {
const ctx = await blankExecuteContext()
ctx.data.link = "http://example.com/"
- assertOk(generatedDownloadFileName(ctx, await buildDownloadableURL(ctx)))
+ assert.ok(generatedDownloadFileName(ctx, await buildDownloadableURL(ctx)))
})
})
it("buffer to object url", () => {
- assertOk(bufferToObjectURL(new ArrayBuffer(10)))
+ assert.ok(bufferToObjectURL(new ArrayBuffer(10)))
})
it("build url for text", async () => {
let ctx = await blankExecuteContext()
ctx.data.selection = "hello"
- assertOk(buildDownloadableURL(ctx))
+ assert.ok(buildDownloadableURL(ctx))
})
it("build url for image", async () => {
let ctx = await blankExecuteContext()
ctx.data.imageSource = browser.runtime.getURL("icon/drag.png")
- assertOk(buildDownloadableURL(ctx))
+ assert.ok(buildDownloadableURL(ctx))
})
it("build url for link", async () => {
let ctx = await blankExecuteContext()
ctx.data.link = "http://www.example.com"
- assertOk(buildDownloadableURL(ctx))
+ assert.ok(buildDownloadableURL(ctx))
})
})
\ No newline at end of file
diff --git a/src/config/config.test.ts b/src/config/config.test.ts
index fc514db..a9ce249 100644
--- a/src/config/config.test.ts
+++ b/src/config/config.test.ts
@@ -1,9 +1,6 @@
-import { assertOk } from "../utils/test_helper"
+import { assert } from "chai"
import { ActionConfig, BroadcastEventTarget, CommandRequest, configBroadcast as configBroadcast, Configuration, type ReadonlyConfiguration } from "./config"
-import chai from "chai"
-const assert = chai.assert
-
describe("test configuration", () => {
it("empty config", () => {
new Configuration()
@@ -11,14 +8,14 @@ describe("test configuration", () => {
it("action config", () => {
const action = new ActionConfig({})
- assertOk(action.toPlainObject())
+ assert.ok(action.toPlainObject())
})
it("request config", () => {
const req = new CommandRequest({
"url": "http://example.com"
})
- assertOk(req.toPlainObject())
+ assert.ok(req.toPlainObject())
})
it("broadcast", () => {
diff --git a/src/content_scripts/utils.test.ts b/src/content_scripts/utils.test.ts
index e617d59..cfb725a 100644
--- a/src/content_scripts/utils.test.ts
+++ b/src/content_scripts/utils.test.ts
@@ -1,4 +1,4 @@
-import { assertOk } from "../utils/test_helper"
+import { assert } from "chai"
import { getAngle, TinyLRU } from "./utils"
describe("content script utils", () => {
@@ -27,28 +27,28 @@ describe("content script utils", () => {
]
for (const tt of testCases) {
let a = getAngle(tt.args[0], tt.args[1])
- assertOk(tt.angle === a, tt.angle, " != ", a)
+ assert.ok(tt.angle === a)
}
})
it('tiny lru', () => {
const lru = new TinyLRU()
- assertOk(lru.get(1) === undefined)
- assertOk(lru.get(2) === undefined)
- assertOk(lru.get(undefined) === undefined)
+ assert.ok(lru.get(1) === undefined)
+ assert.ok(lru.get(2) === undefined)
+ assert.ok(lru.get(undefined) === undefined)
lru.put(1, 1)
lru.put(2, 2)
- assertOk(lru.get(1) === 1)
- assertOk(lru.get(2) === 2)
+ assert.ok(lru.get(1) === 1)
+ assert.ok(lru.get(2) === 2)
- assertOk(lru.get(0) === undefined)
- assertOk(lru.size() === 2)
+ assert.ok(lru.get(0) === undefined)
+ assert.ok(lru.size() === 2)
lru.clear()
- assertOk(lru.size() === 0)
+ assert.ok(lru.size() === 0)
})
})
\ No newline at end of file
diff --git a/src/utils/test.ts b/src/utils/test.ts
new file mode 100644
index 0000000..9a00a4f
--- /dev/null
+++ b/src/utils/test.ts
@@ -0,0 +1,8 @@
+import browser from 'webextension-polyfill';
+
+export async function closeTab(tabIds: number | number[], delayMs?: number) {
+ if (typeof delayMs === 'number') {
+ await new Promise(r => setTimeout(r, delayMs))
+ }
+ return browser.tabs.remove(tabIds)
+}
\ No newline at end of file
diff --git a/src/utils/test_helper.ts b/src/utils/test_helper.ts
deleted file mode 100644
index 64906c3..0000000
--- a/src/utils/test_helper.ts
+++ /dev/null
@@ -1,22 +0,0 @@
-
-import browser from 'webextension-polyfill';
-import { assert } from 'chai'
-
-export function assertEqual(expected: any, actual: any, ...message: any[]) {
- return assert.deepEqual(actual, expected, ...message)
-}
-
-export function assertOk(value: any, ...message: any[]) {
- return assert.ok(value, ...message)
-}
-
-export function assertFail(value: any, ...message: any[]) {
- return assert.notOk(value, ...message)
-}
-
-export async function closeTab(tabIds: number | number[], delayMs?: number) {
- if (typeof delayMs === 'number') {
- await new Promise(r => setTimeout(r, delayMs))
- }
- return browser.tabs.remove(tabIds)
-}
\ No newline at end of file
diff --git a/src/utils/var_substitute.test.ts b/src/utils/var_substitute.test.ts
index d0ccb5c..f292d5c 100644
--- a/src/utils/var_substitute.test.ts
+++ b/src/utils/var_substitute.test.ts
@@ -1,4 +1,3 @@
-import { assertEqual } from "./test_helper"
import { VarSubstituteTemplate } from "./var_substitute"
import { assert } from 'chai'
From 595845daf035fc22efc100bc83a352c99c8e0537 Mon Sep 17 00:00:00 2001
From: harytfw
Date: Wed, 18 Jan 2023 19:43:07 +0800
Subject: [PATCH 13/21] chore: fix archive file was not generated when enable
lint flag
Signed-off-by: harytfw
---
scripts/cli.mjs | 9 ++++-----
1 file changed, 4 insertions(+), 5 deletions(-)
diff --git a/scripts/cli.mjs b/scripts/cli.mjs
index ddc05ed..e7f425e 100644
--- a/scripts/cli.mjs
+++ b/scripts/cli.mjs
@@ -147,10 +147,6 @@ program.command('build')
});
} else {
await buildWithRollup(config)
-
- if (args.lint) {
- webExt.cmd.lint({ sourceDir: dist })
- }
await zipDir(
dist,
{
@@ -161,7 +157,10 @@ program.command('build')
}
);
}
-
+ if (args.lint) {
+ // linter will terminate program with proper exit code
+ await webExt.cmd.lint({ sourceDir: dist }, { shouldExitProgram: true })
+ }
});
program.command('test')
From 62374fc0a9dd6a3d0d213083713defd5e2ef07fa Mon Sep 17 00:00:00 2001
From: harytfw
Date: Thu, 19 Jan 2023 11:38:33 +0800
Subject: [PATCH 14/21] feat: add feature tab to manage features
Signed-off-by: harytfw
---
src/config/config.ts | 1 +
src/options/common.ts | 12 +++++--
src/options/feature.svelte | 53 +++++++++++++++++++++++++++
src/options/nav.svelte | 73 ++++++++++++++++++++++++++++++--------
src/options/options.svelte | 3 ++
src/options/store.ts | 1 +
6 files changed, 127 insertions(+), 16 deletions(-)
create mode 100644 src/options/feature.svelte
diff --git a/src/config/config.ts b/src/config/config.ts
index 7bd4a7a..fb7d1ea 100644
--- a/src/config/config.ts
+++ b/src/config/config.ts
@@ -84,6 +84,7 @@ export type ContextData = {
};
export enum Feature {
+ featureTab = "featureTab",
middleButtonSelector = "middleButtonSelector",
retainComponent = "retainComponent",
auxClose = "auxClose",
diff --git a/src/options/common.ts b/src/options/common.ts
index 4f41775..95889b0 100644
--- a/src/options/common.ts
+++ b/src/options/common.ts
@@ -1,6 +1,5 @@
-import { CommandKind, Direction, OperationMode, TabPosition, ContextType, ContextDataType, CompatibilityStatus } from "../config/config";
+import { CommandKind, Direction, OperationMode, TabPosition, ContextType, ContextDataType, CompatibilityStatus, Feature } from "../config/config";
import { LocaleMessageHelper } from "../locale";
-import { titleCase } from "./utils";
const localeHelper = new LocaleMessageHelper()
@@ -14,6 +13,7 @@ export enum Tab {
common = "common",
configEditor = "configEditor",
compatibility = "compatibility",
+ feature = "feature",
}
@@ -100,4 +100,12 @@ export const compatibilityStatusOptions: Readonly> = Object.values(Feature)
+ .map(s => {
+ return {
+ label: s as string,
+ value: s,
+ }
})
\ No newline at end of file
diff --git a/src/options/feature.svelte b/src/options/feature.svelte
new file mode 100644
index 0000000..b0568b0
--- /dev/null
+++ b/src/options/feature.svelte
@@ -0,0 +1,53 @@
+
+
+
+
+
+
+
diff --git a/src/options/nav.svelte b/src/options/nav.svelte
index d913f3b..662a4ca 100644
--- a/src/options/nav.svelte
+++ b/src/options/nav.svelte
@@ -1,21 +1,23 @@
-
-
+