mirror of
https://github.com/louislam/uptime-kuma.git
synced 2025-02-25 21:15:55 +00:00
Compare commits
6 commits
db757b7c23
...
cd0f2d14f9
Author | SHA1 | Date | |
---|---|---|---|
|
cd0f2d14f9 | ||
|
20820f5a5a | ||
|
66908c7055 | ||
|
7a9191761d | ||
|
9e2c66657d | ||
|
63add0376e |
13 changed files with 636 additions and 324 deletions
223
package-lock.json
generated
223
package-lock.json
generated
|
@ -1,12 +1,12 @@
|
||||||
{
|
{
|
||||||
"name": "uptime-kuma",
|
"name": "uptime-kuma",
|
||||||
"version": "2.0.0-beta.0",
|
"version": "2.0.0-beta.1",
|
||||||
"lockfileVersion": 3,
|
"lockfileVersion": 3,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "uptime-kuma",
|
"name": "uptime-kuma",
|
||||||
"version": "2.0.0-beta.0",
|
"version": "2.0.0-beta.1",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@grpc/grpc-js": "~1.8.22",
|
"@grpc/grpc-js": "~1.8.22",
|
||||||
|
@ -37,8 +37,8 @@
|
||||||
"html-escaper": "^3.0.3",
|
"html-escaper": "^3.0.3",
|
||||||
"http-cookie-agent": "~5.0.4",
|
"http-cookie-agent": "~5.0.4",
|
||||||
"http-graceful-shutdown": "~3.1.7",
|
"http-graceful-shutdown": "~3.1.7",
|
||||||
"http-proxy-agent": "~5.0.0",
|
"http-proxy-agent": "~7.0.2",
|
||||||
"https-proxy-agent": "~5.0.1",
|
"https-proxy-agent": "~7.0.6",
|
||||||
"iconv-lite": "~0.6.3",
|
"iconv-lite": "~0.6.3",
|
||||||
"isomorphic-ws": "^5.0.0",
|
"isomorphic-ws": "^5.0.0",
|
||||||
"jsesc": "~3.0.2",
|
"jsesc": "~3.0.2",
|
||||||
|
@ -60,7 +60,7 @@
|
||||||
"node-cloudflared-tunnel": "~1.0.9",
|
"node-cloudflared-tunnel": "~1.0.9",
|
||||||
"node-radius-client": "~1.0.0",
|
"node-radius-client": "~1.0.0",
|
||||||
"nodemailer": "~6.9.13",
|
"nodemailer": "~6.9.13",
|
||||||
"nostr-tools": "^1.13.1",
|
"nostr-tools": "^2.10.4",
|
||||||
"notp": "~2.0.3",
|
"notp": "~2.0.3",
|
||||||
"openid-client": "^5.4.2",
|
"openid-client": "^5.4.2",
|
||||||
"password-hash": "~1.2.2",
|
"password-hash": "~1.2.2",
|
||||||
|
@ -77,7 +77,7 @@
|
||||||
"semver": "~7.5.4",
|
"semver": "~7.5.4",
|
||||||
"socket.io": "~4.8.0",
|
"socket.io": "~4.8.0",
|
||||||
"socket.io-client": "~4.8.0",
|
"socket.io-client": "~4.8.0",
|
||||||
"socks-proxy-agent": "6.1.1",
|
"socks-proxy-agent": "~8.0.5",
|
||||||
"tar": "~6.2.1",
|
"tar": "~6.2.1",
|
||||||
"tcp-ping": "~0.1.1",
|
"tcp-ping": "~0.1.1",
|
||||||
"thirty-two": "~1.0.2",
|
"thirty-two": "~1.0.2",
|
||||||
|
@ -1052,32 +1052,6 @@
|
||||||
"node": ">=18.0.0"
|
"node": ">=18.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@azure/core-rest-pipeline/node_modules/http-proxy-agent": {
|
|
||||||
"version": "7.0.2",
|
|
||||||
"resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-7.0.2.tgz",
|
|
||||||
"integrity": "sha512-T1gkAiYYDWYx3V5Bmyu7HcfcvL7mUrTWiM6yOfa3PIphViJ/gFPbvidQ+veqSOHci/PxBcDabeUNCzpOODJZig==",
|
|
||||||
"license": "MIT",
|
|
||||||
"dependencies": {
|
|
||||||
"agent-base": "^7.1.0",
|
|
||||||
"debug": "^4.3.4"
|
|
||||||
},
|
|
||||||
"engines": {
|
|
||||||
"node": ">= 14"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"node_modules/@azure/core-rest-pipeline/node_modules/https-proxy-agent": {
|
|
||||||
"version": "7.0.6",
|
|
||||||
"resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-7.0.6.tgz",
|
|
||||||
"integrity": "sha512-vK9P5/iUfdl95AI+JVyUuIcVtd4ofvtrOr3HNtM2yxC9bnMbEdp3x01OhQNnjb8IJYi38VlTE3mBXwcfvywuSw==",
|
|
||||||
"license": "MIT",
|
|
||||||
"dependencies": {
|
|
||||||
"agent-base": "^7.1.2",
|
|
||||||
"debug": "4"
|
|
||||||
},
|
|
||||||
"engines": {
|
|
||||||
"node": ">= 14"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"node_modules/@azure/core-tracing": {
|
"node_modules/@azure/core-tracing": {
|
||||||
"version": "1.2.0",
|
"version": "1.2.0",
|
||||||
"resolved": "https://registry.npmjs.org/@azure/core-tracing/-/core-tracing-1.2.0.tgz",
|
"resolved": "https://registry.npmjs.org/@azure/core-tracing/-/core-tracing-1.2.0.tgz",
|
||||||
|
@ -2827,6 +2801,29 @@
|
||||||
"node-pre-gyp": "bin/node-pre-gyp"
|
"node-pre-gyp": "bin/node-pre-gyp"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@mapbox/node-pre-gyp/node_modules/agent-base": {
|
||||||
|
"version": "6.0.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz",
|
||||||
|
"integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==",
|
||||||
|
"dependencies": {
|
||||||
|
"debug": "4"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 6.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/@mapbox/node-pre-gyp/node_modules/https-proxy-agent": {
|
||||||
|
"version": "5.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.1.tgz",
|
||||||
|
"integrity": "sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==",
|
||||||
|
"dependencies": {
|
||||||
|
"agent-base": "6",
|
||||||
|
"debug": "4"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 6"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@mongodb-js/saslprep": {
|
"node_modules/@mongodb-js/saslprep": {
|
||||||
"version": "1.1.9",
|
"version": "1.1.9",
|
||||||
"resolved": "https://registry.npmjs.org/@mongodb-js/saslprep/-/saslprep-1.1.9.tgz",
|
"resolved": "https://registry.npmjs.org/@mongodb-js/saslprep/-/saslprep-1.1.9.tgz",
|
||||||
|
@ -2838,9 +2835,9 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@noble/ciphers": {
|
"node_modules/@noble/ciphers": {
|
||||||
"version": "0.2.0",
|
"version": "0.5.3",
|
||||||
"resolved": "https://registry.npmjs.org/@noble/ciphers/-/ciphers-0.2.0.tgz",
|
"resolved": "https://registry.npmjs.org/@noble/ciphers/-/ciphers-0.5.3.tgz",
|
||||||
"integrity": "sha512-6YBxJDAapHSdd3bLDv6x2wRPwq4QFMUaB3HvljNBUTThDd12eSm7/3F+2lnfzx2jvM+S6Nsy0jEt9QbPqSwqRw==",
|
"integrity": "sha512-B0+6IIHiqEs3BPMT0hcRmHvEj2QHOLu+uwt+tqDDeVd0oyVzh7BPrDcPjRnV1PV/5LaknXJJQvOuRGR0zQJz+w==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"funding": {
|
"funding": {
|
||||||
"url": "https://paulmillr.com/funding/"
|
"url": "https://paulmillr.com/funding/"
|
||||||
|
@ -4270,15 +4267,6 @@
|
||||||
"testcontainers": "^10.16.0"
|
"testcontainers": "^10.16.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@tootallnate/once": {
|
|
||||||
"version": "2.0.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/@tootallnate/once/-/once-2.0.0.tgz",
|
|
||||||
"integrity": "sha512-XCuKFP5PS55gnMVu3dty8KPatLqUoy/ZYzDzAGCQ8JNFCkLXzmI7vNHCR+XpbZaMWQK/vQubr7PkYq8g470J/A==",
|
|
||||||
"license": "MIT",
|
|
||||||
"engines": {
|
|
||||||
"node": ">= 10"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"node_modules/@types/accepts": {
|
"node_modules/@types/accepts": {
|
||||||
"version": "1.3.7",
|
"version": "1.3.7",
|
||||||
"resolved": "https://registry.npmjs.org/@types/accepts/-/accepts-1.3.7.tgz",
|
"resolved": "https://registry.npmjs.org/@types/accepts/-/accepts-1.3.7.tgz",
|
||||||
|
@ -10117,29 +10105,15 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/http-proxy-agent": {
|
"node_modules/http-proxy-agent": {
|
||||||
"version": "5.0.0",
|
"version": "7.0.2",
|
||||||
"resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-5.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-7.0.2.tgz",
|
||||||
"integrity": "sha512-n2hY8YdoRE1i7r6M0w9DIw5GgZN0G25P8zLCRQ8rjXtTU3vsNFBI/vWK/UIeE6g5MUUz6avwAPXmL6Fy9D/90w==",
|
"integrity": "sha512-T1gkAiYYDWYx3V5Bmyu7HcfcvL7mUrTWiM6yOfa3PIphViJ/gFPbvidQ+veqSOHci/PxBcDabeUNCzpOODJZig==",
|
||||||
"license": "MIT",
|
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@tootallnate/once": "2",
|
"agent-base": "^7.1.0",
|
||||||
"agent-base": "6",
|
"debug": "^4.3.4"
|
||||||
"debug": "4"
|
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">= 6"
|
"node": ">= 14"
|
||||||
}
|
|
||||||
},
|
|
||||||
"node_modules/http-proxy-agent/node_modules/agent-base": {
|
|
||||||
"version": "6.0.2",
|
|
||||||
"resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz",
|
|
||||||
"integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==",
|
|
||||||
"license": "MIT",
|
|
||||||
"dependencies": {
|
|
||||||
"debug": "4"
|
|
||||||
},
|
|
||||||
"engines": {
|
|
||||||
"node": ">= 6.0.0"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/http2-wrapper": {
|
"node_modules/http2-wrapper": {
|
||||||
|
@ -10156,28 +10130,15 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/https-proxy-agent": {
|
"node_modules/https-proxy-agent": {
|
||||||
"version": "5.0.1",
|
"version": "7.0.6",
|
||||||
"resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-7.0.6.tgz",
|
||||||
"integrity": "sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==",
|
"integrity": "sha512-vK9P5/iUfdl95AI+JVyUuIcVtd4ofvtrOr3HNtM2yxC9bnMbEdp3x01OhQNnjb8IJYi38VlTE3mBXwcfvywuSw==",
|
||||||
"license": "MIT",
|
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"agent-base": "6",
|
"agent-base": "^7.1.2",
|
||||||
"debug": "4"
|
"debug": "4"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">= 6"
|
"node": ">= 14"
|
||||||
}
|
|
||||||
},
|
|
||||||
"node_modules/https-proxy-agent/node_modules/agent-base": {
|
|
||||||
"version": "6.0.2",
|
|
||||||
"resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz",
|
|
||||||
"integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==",
|
|
||||||
"license": "MIT",
|
|
||||||
"dependencies": {
|
|
||||||
"debug": "4"
|
|
||||||
},
|
|
||||||
"engines": {
|
|
||||||
"node": ">= 6.0.0"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/human-signals": {
|
"node_modules/human-signals": {
|
||||||
|
@ -11679,6 +11640,33 @@
|
||||||
"node": ">= 6"
|
"node": ">= 6"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/make-fetch-happen/node_modules/https-proxy-agent": {
|
||||||
|
"version": "5.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.1.tgz",
|
||||||
|
"integrity": "sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==",
|
||||||
|
"optional": true,
|
||||||
|
"dependencies": {
|
||||||
|
"agent-base": "6",
|
||||||
|
"debug": "4"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 6"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/make-fetch-happen/node_modules/socks-proxy-agent": {
|
||||||
|
"version": "6.2.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/socks-proxy-agent/-/socks-proxy-agent-6.2.1.tgz",
|
||||||
|
"integrity": "sha512-a6KW9G+6B3nWZ1yB8G7pJwL3ggLy1uTzKAgCb7ttblwqdz9fMGJUuTy3uFzEP48FAs9FLILlmzDlE2JJhVQaXQ==",
|
||||||
|
"optional": true,
|
||||||
|
"dependencies": {
|
||||||
|
"agent-base": "^6.0.2",
|
||||||
|
"debug": "^4.3.3",
|
||||||
|
"socks": "^2.6.2"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 10"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/map-obj": {
|
"node_modules/map-obj": {
|
||||||
"version": "4.3.0",
|
"version": "4.3.0",
|
||||||
"resolved": "https://registry.npmjs.org/map-obj/-/map-obj-4.3.0.tgz",
|
"resolved": "https://registry.npmjs.org/map-obj/-/map-obj-4.3.0.tgz",
|
||||||
|
@ -12582,18 +12570,21 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/nostr-tools": {
|
"node_modules/nostr-tools": {
|
||||||
"version": "1.17.0",
|
"version": "2.10.4",
|
||||||
"resolved": "https://registry.npmjs.org/nostr-tools/-/nostr-tools-1.17.0.tgz",
|
"resolved": "https://registry.npmjs.org/nostr-tools/-/nostr-tools-2.10.4.tgz",
|
||||||
"integrity": "sha512-LZmR8GEWKZeElbFV5Xte75dOeE9EFUW/QLI1Ncn3JKn0kFddDKEfBbFN8Mu4TMs+L4HR/WTPha2l+PPuRnJcMw==",
|
"integrity": "sha512-biU7sk+jxHgVASfobg2T5ttxOGGSt69wEVBC51sHHOEaKAAdzHBLV/I2l9Rf61UzClhliZwNouYhqIso4a3HYg==",
|
||||||
"license": "Unlicense",
|
"license": "Unlicense",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@noble/ciphers": "0.2.0",
|
"@noble/ciphers": "^0.5.1",
|
||||||
"@noble/curves": "1.1.0",
|
"@noble/curves": "1.2.0",
|
||||||
"@noble/hashes": "1.3.1",
|
"@noble/hashes": "1.3.1",
|
||||||
"@scure/base": "1.1.1",
|
"@scure/base": "1.1.1",
|
||||||
"@scure/bip32": "1.3.1",
|
"@scure/bip32": "1.3.1",
|
||||||
"@scure/bip39": "1.2.1"
|
"@scure/bip39": "1.2.1"
|
||||||
},
|
},
|
||||||
|
"optionalDependencies": {
|
||||||
|
"nostr-wasm": "0.1.0"
|
||||||
|
},
|
||||||
"peerDependencies": {
|
"peerDependencies": {
|
||||||
"typescript": ">=5.0.0"
|
"typescript": ">=5.0.0"
|
||||||
},
|
},
|
||||||
|
@ -12603,6 +12594,37 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/nostr-tools/node_modules/@noble/curves": {
|
||||||
|
"version": "1.2.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/@noble/curves/-/curves-1.2.0.tgz",
|
||||||
|
"integrity": "sha512-oYclrNgRaM9SsBUBVbb8M6DTV7ZHRTKugureoYEncY5c65HOmRzvSiTE3y5CYaPYJA/GVkrhXEoF0M3Ya9PMnw==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@noble/hashes": "1.3.2"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://paulmillr.com/funding/"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/nostr-tools/node_modules/@noble/curves/node_modules/@noble/hashes": {
|
||||||
|
"version": "1.3.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/@noble/hashes/-/hashes-1.3.2.tgz",
|
||||||
|
"integrity": "sha512-MVC8EAQp7MvEcm30KWENFjgR+Mkmf+D189XJTkFIlwohU5hcBbn1ZkKq7KVTi2Hme3PMGF390DaL52beVrIihQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 16"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://paulmillr.com/funding/"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/nostr-wasm": {
|
||||||
|
"version": "0.1.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/nostr-wasm/-/nostr-wasm-0.1.0.tgz",
|
||||||
|
"integrity": "sha512-78BTryCLcLYv96ONU8Ws3Q1JzjlAt+43pWQhIl86xZmWeegYCNLPml7yQ+gG3vR6V5h4XGj+TxO+SS5dsThQIA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"optional": true
|
||||||
|
},
|
||||||
"node_modules/notp": {
|
"node_modules/notp": {
|
||||||
"version": "2.0.3",
|
"version": "2.0.3",
|
||||||
"resolved": "https://registry.npmjs.org/notp/-/notp-2.0.3.tgz",
|
"resolved": "https://registry.npmjs.org/notp/-/notp-2.0.3.tgz",
|
||||||
|
@ -15227,29 +15249,16 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/socks-proxy-agent": {
|
"node_modules/socks-proxy-agent": {
|
||||||
"version": "6.1.1",
|
"version": "8.0.5",
|
||||||
"resolved": "https://registry.npmjs.org/socks-proxy-agent/-/socks-proxy-agent-6.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/socks-proxy-agent/-/socks-proxy-agent-8.0.5.tgz",
|
||||||
"integrity": "sha512-t8J0kG3csjA4g6FTbsMOWws+7R7vuRC8aQ/wy3/1OWmsgwA68zs/+cExQ0koSitUDXqhufF/YJr9wtNMZHw5Ew==",
|
"integrity": "sha512-HehCEsotFqbPW9sJ8WVYB6UbmIMv7kUUORIF2Nncq4VQvBfNBLibW9YZR5dlYCSUhwcD628pRllm7n+E+YTzJw==",
|
||||||
"license": "MIT",
|
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"agent-base": "^6.0.2",
|
"agent-base": "^7.1.2",
|
||||||
"debug": "^4.3.1",
|
"debug": "^4.3.4",
|
||||||
"socks": "^2.6.1"
|
"socks": "^2.8.3"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">= 10"
|
"node": ">= 14"
|
||||||
}
|
|
||||||
},
|
|
||||||
"node_modules/socks-proxy-agent/node_modules/agent-base": {
|
|
||||||
"version": "6.0.2",
|
|
||||||
"resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz",
|
|
||||||
"integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==",
|
|
||||||
"license": "MIT",
|
|
||||||
"dependencies": {
|
|
||||||
"debug": "4"
|
|
||||||
},
|
|
||||||
"engines": {
|
|
||||||
"node": ">= 6.0.0"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/sortablejs": {
|
"node_modules/sortablejs": {
|
||||||
|
|
|
@ -95,8 +95,8 @@
|
||||||
"html-escaper": "^3.0.3",
|
"html-escaper": "^3.0.3",
|
||||||
"http-cookie-agent": "~5.0.4",
|
"http-cookie-agent": "~5.0.4",
|
||||||
"http-graceful-shutdown": "~3.1.7",
|
"http-graceful-shutdown": "~3.1.7",
|
||||||
"http-proxy-agent": "~5.0.0",
|
"http-proxy-agent": "~7.0.2",
|
||||||
"https-proxy-agent": "~5.0.1",
|
"https-proxy-agent": "~7.0.6",
|
||||||
"iconv-lite": "~0.6.3",
|
"iconv-lite": "~0.6.3",
|
||||||
"isomorphic-ws": "^5.0.0",
|
"isomorphic-ws": "^5.0.0",
|
||||||
"jsesc": "~3.0.2",
|
"jsesc": "~3.0.2",
|
||||||
|
@ -118,7 +118,7 @@
|
||||||
"node-cloudflared-tunnel": "~1.0.9",
|
"node-cloudflared-tunnel": "~1.0.9",
|
||||||
"node-radius-client": "~1.0.0",
|
"node-radius-client": "~1.0.0",
|
||||||
"nodemailer": "~6.9.13",
|
"nodemailer": "~6.9.13",
|
||||||
"nostr-tools": "^1.13.1",
|
"nostr-tools": "^2.10.4",
|
||||||
"notp": "~2.0.3",
|
"notp": "~2.0.3",
|
||||||
"openid-client": "^5.4.2",
|
"openid-client": "^5.4.2",
|
||||||
"password-hash": "~1.2.2",
|
"password-hash": "~1.2.2",
|
||||||
|
@ -135,7 +135,7 @@
|
||||||
"semver": "~7.5.4",
|
"semver": "~7.5.4",
|
||||||
"socket.io": "~4.8.0",
|
"socket.io": "~4.8.0",
|
||||||
"socket.io-client": "~4.8.0",
|
"socket.io-client": "~4.8.0",
|
||||||
"socks-proxy-agent": "6.1.1",
|
"socks-proxy-agent": "~8.0.5",
|
||||||
"tar": "~6.2.1",
|
"tar": "~6.2.1",
|
||||||
"tcp-ping": "~0.1.1",
|
"tcp-ping": "~0.1.1",
|
||||||
"thirty-two": "~1.0.2",
|
"thirty-two": "~1.0.2",
|
||||||
|
|
|
@ -1,11 +1,10 @@
|
||||||
const NotificationProvider = require("./notification-provider");
|
const NotificationProvider = require("./notification-provider");
|
||||||
const {
|
const {
|
||||||
relayInit,
|
finalizeEvent,
|
||||||
getPublicKey,
|
Relay,
|
||||||
getEventHash,
|
kinds,
|
||||||
getSignature,
|
|
||||||
nip04,
|
nip04,
|
||||||
nip19
|
nip19,
|
||||||
} = require("nostr-tools");
|
} = require("nostr-tools");
|
||||||
|
|
||||||
// polyfills for node versions
|
// polyfills for node versions
|
||||||
|
@ -31,7 +30,6 @@ class Nostr extends NotificationProvider {
|
||||||
const createdAt = Math.floor(Date.now() / 1000);
|
const createdAt = Math.floor(Date.now() / 1000);
|
||||||
|
|
||||||
const senderPrivateKey = await this.getPrivateKey(notification.sender);
|
const senderPrivateKey = await this.getPrivateKey(notification.sender);
|
||||||
const senderPublicKey = getPublicKey(senderPrivateKey);
|
|
||||||
const recipientsPublicKeys = await this.getPublicKeys(notification.recipients);
|
const recipientsPublicKeys = await this.getPublicKeys(notification.recipients);
|
||||||
|
|
||||||
// Create NIP-04 encrypted direct message event for each recipient
|
// Create NIP-04 encrypted direct message event for each recipient
|
||||||
|
@ -39,34 +37,41 @@ class Nostr extends NotificationProvider {
|
||||||
for (const recipientPublicKey of recipientsPublicKeys) {
|
for (const recipientPublicKey of recipientsPublicKeys) {
|
||||||
const ciphertext = await nip04.encrypt(senderPrivateKey, recipientPublicKey, msg);
|
const ciphertext = await nip04.encrypt(senderPrivateKey, recipientPublicKey, msg);
|
||||||
let event = {
|
let event = {
|
||||||
kind: 4,
|
kind: kinds.EncryptedDirectMessage,
|
||||||
pubkey: senderPublicKey,
|
|
||||||
created_at: createdAt,
|
created_at: createdAt,
|
||||||
tags: [[ "p", recipientPublicKey ]],
|
tags: [[ "p", recipientPublicKey ]],
|
||||||
content: ciphertext,
|
content: ciphertext,
|
||||||
};
|
};
|
||||||
event.id = getEventHash(event);
|
const signedEvent = finalizeEvent(event, senderPrivateKey);
|
||||||
event.sig = getSignature(event, senderPrivateKey);
|
events.push(signedEvent);
|
||||||
events.push(event);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Publish events to each relay
|
// Publish events to each relay
|
||||||
const relays = notification.relays.split("\n");
|
const relays = notification.relays.split("\n");
|
||||||
let successfulRelays = 0;
|
let successfulRelays = 0;
|
||||||
|
|
||||||
// Connect to each relay
|
|
||||||
for (const relayUrl of relays) {
|
for (const relayUrl of relays) {
|
||||||
const relay = relayInit(relayUrl);
|
const relay = await Relay.connect(relayUrl);
|
||||||
try {
|
let eventIndex = 0;
|
||||||
await relay.connect();
|
|
||||||
successfulRelays++;
|
|
||||||
|
|
||||||
// Publish events
|
// Authenticate to the relay, if required
|
||||||
for (const event of events) {
|
try {
|
||||||
relay.publish(event);
|
await relay.publish(events[0]);
|
||||||
}
|
eventIndex = 1;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
continue;
|
if (relay.challenge) {
|
||||||
|
await relay.auth(async (evt) => {
|
||||||
|
return finalizeEvent(evt, senderPrivateKey);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
for (let i = eventIndex; i < events.length; i++) {
|
||||||
|
await relay.publish(events[i]);
|
||||||
|
}
|
||||||
|
successfulRelays++;
|
||||||
|
} catch (error) {
|
||||||
|
console.error(`Failed to publish event to ${relayUrl}:`, error);
|
||||||
} finally {
|
} finally {
|
||||||
relay.close();
|
relay.close();
|
||||||
}
|
}
|
||||||
|
@ -90,7 +95,7 @@ class Nostr extends NotificationProvider {
|
||||||
const { data } = senderDecodeResult;
|
const { data } = senderDecodeResult;
|
||||||
return data;
|
return data;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
throw new Error(`Failed to get private key: ${error.message}`);
|
throw new Error(`Failed to decode private key for sender ${sender}: ${error.message}`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -109,10 +114,10 @@ class Nostr extends NotificationProvider {
|
||||||
if (type === "npub") {
|
if (type === "npub") {
|
||||||
publicKeys.push(data);
|
publicKeys.push(data);
|
||||||
} else {
|
} else {
|
||||||
throw new Error("not an npub");
|
throw new Error(`Recipient ${recipient} is not an npub`);
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
throw new Error(`Error decoding recipient: ${error}`);
|
throw new Error(`Error decoding recipient ${recipient}: ${error}`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return publicKeys;
|
return publicKeys;
|
||||||
|
|
|
@ -11,7 +11,8 @@ class PushDeer extends NotificationProvider {
|
||||||
async send(notification, msg, monitorJSON = null, heartbeatJSON = null) {
|
async send(notification, msg, monitorJSON = null, heartbeatJSON = null) {
|
||||||
const okMsg = "Sent Successfully.";
|
const okMsg = "Sent Successfully.";
|
||||||
const serverUrl = notification.pushdeerServer || "https://api2.pushdeer.com";
|
const serverUrl = notification.pushdeerServer || "https://api2.pushdeer.com";
|
||||||
const url = `${serverUrl.trim().replace(/\/*$/, "")}/message/push`;
|
// capture group below is nessesary to prevent an ReDOS-attack
|
||||||
|
const url = `${serverUrl.trim().replace(/([^/])\/+$/, "$1")}/message/push`;
|
||||||
|
|
||||||
let valid = msg != null && monitorJSON != null && heartbeatJSON != null;
|
let valid = msg != null && monitorJSON != null && heartbeatJSON != null;
|
||||||
|
|
||||||
|
|
|
@ -24,7 +24,7 @@ class Whapi extends NotificationProvider {
|
||||||
"body": msg,
|
"body": msg,
|
||||||
};
|
};
|
||||||
|
|
||||||
let url = (notification.whapiApiUrl || "https://gate.whapi.cloud/").replace(/\/+$/, "") + "/messages/text";
|
let url = (notification.whapiApiUrl || "https://gate.whapi.cloud/").replace(/([^/])\/+$/, "$1") + "/messages/text";
|
||||||
|
|
||||||
await axios.post(url, data, config);
|
await axios.post(url, data, config);
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
const { R } = require("redbean-node");
|
const { R } = require("redbean-node");
|
||||||
const HttpProxyAgent = require("http-proxy-agent");
|
const { HttpProxyAgent } = require("http-proxy-agent");
|
||||||
const HttpsProxyAgent = require("https-proxy-agent");
|
const { HttpsProxyAgent } = require("https-proxy-agent");
|
||||||
const SocksProxyAgent = require("socks-proxy-agent");
|
const { SocksProxyAgent } = require("socks-proxy-agent");
|
||||||
const { debug } = require("../src/util");
|
const { debug } = require("../src/util");
|
||||||
const { UptimeKumaServer } = require("./uptime-kuma-server");
|
const { UptimeKumaServer } = require("./uptime-kuma-server");
|
||||||
const { CookieJar } = require("tough-cookie");
|
const { CookieJar } = require("tough-cookie");
|
||||||
|
@ -100,17 +100,17 @@ class Proxy {
|
||||||
let jar = new CookieJar();
|
let jar = new CookieJar();
|
||||||
|
|
||||||
const proxyOptions = {
|
const proxyOptions = {
|
||||||
protocol: proxy.protocol,
|
|
||||||
host: proxy.host,
|
|
||||||
port: proxy.port,
|
|
||||||
cookies: { jar },
|
cookies: { jar },
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const proxyUrl = new URL(`${proxy.protocol}://${proxy.host}:${proxy.port}`);
|
||||||
|
|
||||||
if (proxy.auth) {
|
if (proxy.auth) {
|
||||||
proxyOptions.auth = `${proxy.username}:${proxy.password}`;
|
proxyUrl.username = proxy.username;
|
||||||
|
proxyUrl.password = proxy.password;
|
||||||
}
|
}
|
||||||
|
|
||||||
debug(`Proxy Options: ${JSON.stringify(proxyOptions)}`);
|
debug(`Proxy URL: ${proxyUrl.toString()}`);
|
||||||
debug(`HTTP Agent Options: ${JSON.stringify(httpAgentOptions)}`);
|
debug(`HTTP Agent Options: ${JSON.stringify(httpAgentOptions)}`);
|
||||||
debug(`HTTPS Agent Options: ${JSON.stringify(httpsAgentOptions)}`);
|
debug(`HTTPS Agent Options: ${JSON.stringify(httpsAgentOptions)}`);
|
||||||
|
|
||||||
|
@ -122,15 +122,15 @@ class Proxy {
|
||||||
// eslint-disable-next-line no-case-declarations
|
// eslint-disable-next-line no-case-declarations
|
||||||
const HttpsCookieProxyAgent = createCookieAgent(HttpsProxyAgent);
|
const HttpsCookieProxyAgent = createCookieAgent(HttpsProxyAgent);
|
||||||
|
|
||||||
httpAgent = new HttpCookieProxyAgent({
|
httpAgent = new HttpCookieProxyAgent(proxyUrl.toString(), {
|
||||||
...httpAgentOptions || {},
|
...(httpAgentOptions || {}),
|
||||||
|
...proxyOptions,
|
||||||
|
});
|
||||||
|
httpsAgent = new HttpsCookieProxyAgent(proxyUrl.toString(), {
|
||||||
|
...(httpsAgentOptions || {}),
|
||||||
...proxyOptions,
|
...proxyOptions,
|
||||||
});
|
});
|
||||||
|
|
||||||
httpsAgent = new HttpsCookieProxyAgent({
|
|
||||||
...httpsAgentOptions || {},
|
|
||||||
...proxyOptions,
|
|
||||||
});
|
|
||||||
break;
|
break;
|
||||||
case "socks":
|
case "socks":
|
||||||
case "socks5":
|
case "socks5":
|
||||||
|
@ -138,10 +138,9 @@ class Proxy {
|
||||||
case "socks4":
|
case "socks4":
|
||||||
// eslint-disable-next-line no-case-declarations
|
// eslint-disable-next-line no-case-declarations
|
||||||
const SocksCookieProxyAgent = createCookieAgent(SocksProxyAgent);
|
const SocksCookieProxyAgent = createCookieAgent(SocksProxyAgent);
|
||||||
agent = new SocksCookieProxyAgent({
|
agent = new SocksCookieProxyAgent(proxyUrl.toString(), {
|
||||||
...httpAgentOptions,
|
...httpAgentOptions,
|
||||||
...httpsAgentOptions,
|
...httpsAgentOptions,
|
||||||
...proxyOptions,
|
|
||||||
tls: {
|
tls: {
|
||||||
rejectUnauthorized: httpsAgentOptions.rejectUnauthorized,
|
rejectUnauthorized: httpsAgentOptions.rejectUnauthorized,
|
||||||
},
|
},
|
||||||
|
|
|
@ -313,6 +313,9 @@ let needSetup = false;
|
||||||
const statusPageRouter = require("./routers/status-page-router");
|
const statusPageRouter = require("./routers/status-page-router");
|
||||||
app.use(statusPageRouter);
|
app.use(statusPageRouter);
|
||||||
|
|
||||||
|
const { injectDefaultAppearance } = require("./utils/inject-default-appearance");
|
||||||
|
app.use(injectDefaultAppearance);
|
||||||
|
|
||||||
// Universal Route Handler, must be at the end of all express routes.
|
// Universal Route Handler, must be at the end of all express routes.
|
||||||
app.get("*", async (_request, response) => {
|
app.get("*", async (_request, response) => {
|
||||||
if (_request.originalUrl.startsWith("/upload/")) {
|
if (_request.originalUrl.startsWith("/upload/")) {
|
||||||
|
|
50
server/utils/inject-default-appearance.js
Normal file
50
server/utils/inject-default-appearance.js
Normal file
|
@ -0,0 +1,50 @@
|
||||||
|
const { Settings } = require("../settings");
|
||||||
|
const cheerio = require("cheerio");
|
||||||
|
const jsesc = require("jsesc");
|
||||||
|
const { log } = require("../../src/util");
|
||||||
|
|
||||||
|
const injectDefaultAppearance = (req, res, next) => {
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Intercept send() calls and inject Default Appearance
|
||||||
|
// https://stackoverflow.com/a/60817116
|
||||||
|
const oldSend = res.send;
|
||||||
|
res.send = async (data) => {
|
||||||
|
|
||||||
|
if (typeof data === "string") {
|
||||||
|
log.debug("inject-default-appearance", req.method + " " + req.url);
|
||||||
|
const $ = cheerio.load(data);
|
||||||
|
|
||||||
|
const defaultAppearance = await Settings.get("defaultAppearance");
|
||||||
|
if (defaultAppearance) {
|
||||||
|
const head = $("head");
|
||||||
|
|
||||||
|
const escapedJSONObject = jsesc(defaultAppearance, { isScriptContext: true });
|
||||||
|
|
||||||
|
const script = $(`
|
||||||
|
<script id="default-appearance" data-json="{}">
|
||||||
|
window.defaultAppearance = ${escapedJSONObject};
|
||||||
|
</script>
|
||||||
|
`);
|
||||||
|
|
||||||
|
head.append(script);
|
||||||
|
|
||||||
|
data = $.root().html();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
res.send = oldSend; // set function back to avoid 'double-send'
|
||||||
|
return res.send(data);
|
||||||
|
};
|
||||||
|
|
||||||
|
next();
|
||||||
|
} catch (e) {
|
||||||
|
|
||||||
|
next(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
injectDefaultAppearance
|
||||||
|
};
|
73
src/components/HorizontalTabHeader.vue
Normal file
73
src/components/HorizontalTabHeader.vue
Normal file
|
@ -0,0 +1,73 @@
|
||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<ul class="nav nav-tabs">
|
||||||
|
<li v-for="(tab, index) in tabs" :key="index" class="nav-item">
|
||||||
|
<a
|
||||||
|
class="nav-link"
|
||||||
|
:class="{ active: index == selected }"
|
||||||
|
href="#"
|
||||||
|
@click="$emit('update:selected', index)"
|
||||||
|
>
|
||||||
|
{{ tab }}
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="js">
|
||||||
|
export default {
|
||||||
|
props: {
|
||||||
|
tabs: {
|
||||||
|
type: Array[String],
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
selected: {
|
||||||
|
type: Number,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
emits: [ "update:selected" ],
|
||||||
|
data() {
|
||||||
|
return {};
|
||||||
|
},
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
@import "../assets/vars.scss";
|
||||||
|
|
||||||
|
.nav-tabs {
|
||||||
|
border-bottom: 1px solid $primary;
|
||||||
|
|
||||||
|
.nav-item {
|
||||||
|
flex-grow: 1;
|
||||||
|
|
||||||
|
.nav-link {
|
||||||
|
text-align: center;
|
||||||
|
border-top-left-radius: 10px;
|
||||||
|
border-top-right-radius: 10px;
|
||||||
|
|
||||||
|
&.active {
|
||||||
|
background-color: $highlight-white;
|
||||||
|
|
||||||
|
.dark & {
|
||||||
|
color: $dark-font-color;
|
||||||
|
background-color: $dark-header-bg;
|
||||||
|
}
|
||||||
|
|
||||||
|
border-color: transparent transparent $primary transparent;
|
||||||
|
border-width: 1px 1px 6px 1px;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
.dark & {
|
||||||
|
background-color: $dark-header-bg;
|
||||||
|
}
|
||||||
|
border-color: $primary;
|
||||||
|
border-width: 1px 1px 6px 1px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -1,186 +1,97 @@
|
||||||
<template>
|
<template>
|
||||||
<div>
|
<div>
|
||||||
<div class="my-4">
|
<HorizontalTabHeader
|
||||||
<label for="language" class="form-label">
|
v-model:selected="storageLoc"
|
||||||
{{ $t("Language") }}
|
class="mt-4"
|
||||||
</label>
|
:tabs="['Browser', 'Server']"
|
||||||
<select id="language" v-model="$root.language" class="form-select">
|
/>
|
||||||
<option
|
<AppearanceSettings
|
||||||
v-for="(lang, i) in $i18n.availableLocales"
|
v-if="storageLoc == 0"
|
||||||
:key="`Lang${i}`"
|
v-model:language="$root.language"
|
||||||
:value="lang"
|
v-model:userTheme="$root.userTheme"
|
||||||
>
|
v-model:userHeartbeatBar="$root.userHeartbeatBar"
|
||||||
{{ $i18n.messages[lang].languageName }}
|
v-model:styleElapsedTime="$root.styleElapsedTime"
|
||||||
</option>
|
:languages="languagesList"
|
||||||
</select>
|
/>
|
||||||
</div>
|
<AppearanceSettings
|
||||||
<div class="my-4">
|
v-if="storageLoc == 1"
|
||||||
<label for="timezone" class="form-label">{{ $t("Theme") }}</label>
|
:language="settings.defaultAppearance.language || ''"
|
||||||
<div>
|
:userTheme="settings.defaultAppearance.theme || ''"
|
||||||
<div
|
:userHeartbeatBar="settings.defaultAppearance.heartbeatBarTheme || ''"
|
||||||
class="btn-group"
|
:styleElapsedTime="settings.defaultAppearance.styleElapsedTime || ''"
|
||||||
role="group"
|
:languages="languagesList"
|
||||||
aria-label="Basic checkbox toggle button group"
|
@update:language="updateDefaultLanguage"
|
||||||
>
|
@update:user-theme="updateDefaultTheme"
|
||||||
<input
|
@update:user-heartbeat-bar="updateDefaultHeartbeatBarTheme"
|
||||||
id="btncheck1"
|
@update:style-elapsed-time="updateDefaultStyleElapsedTime"
|
||||||
v-model="$root.userTheme"
|
/>
|
||||||
type="radio"
|
|
||||||
class="btn-check"
|
|
||||||
name="theme"
|
|
||||||
autocomplete="off"
|
|
||||||
value="light"
|
|
||||||
/>
|
|
||||||
<label class="btn btn-outline-primary" for="btncheck1">
|
|
||||||
{{ $t("Light") }}
|
|
||||||
</label>
|
|
||||||
|
|
||||||
<input
|
|
||||||
id="btncheck2"
|
|
||||||
v-model="$root.userTheme"
|
|
||||||
type="radio"
|
|
||||||
class="btn-check"
|
|
||||||
name="theme"
|
|
||||||
autocomplete="off"
|
|
||||||
value="dark"
|
|
||||||
/>
|
|
||||||
<label class="btn btn-outline-primary" for="btncheck2">
|
|
||||||
{{ $t("Dark") }}
|
|
||||||
</label>
|
|
||||||
|
|
||||||
<input
|
|
||||||
id="btncheck3"
|
|
||||||
v-model="$root.userTheme"
|
|
||||||
type="radio"
|
|
||||||
class="btn-check"
|
|
||||||
name="theme"
|
|
||||||
autocomplete="off"
|
|
||||||
value="auto"
|
|
||||||
/>
|
|
||||||
<label class="btn btn-outline-primary" for="btncheck3">
|
|
||||||
{{ $t("Auto") }}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="my-4">
|
|
||||||
<label class="form-label">{{ $t("Theme - Heartbeat Bar") }}</label>
|
|
||||||
<div>
|
|
||||||
<div
|
|
||||||
class="btn-group"
|
|
||||||
role="group"
|
|
||||||
aria-label="Basic checkbox toggle button group"
|
|
||||||
>
|
|
||||||
<input
|
|
||||||
id="btncheck4"
|
|
||||||
v-model="$root.userHeartbeatBar"
|
|
||||||
type="radio"
|
|
||||||
class="btn-check"
|
|
||||||
name="heartbeatBarTheme"
|
|
||||||
autocomplete="off"
|
|
||||||
value="normal"
|
|
||||||
/>
|
|
||||||
<label class="btn btn-outline-primary" for="btncheck4">
|
|
||||||
{{ $t("Normal") }}
|
|
||||||
</label>
|
|
||||||
|
|
||||||
<input
|
|
||||||
id="btncheck5"
|
|
||||||
v-model="$root.userHeartbeatBar"
|
|
||||||
type="radio"
|
|
||||||
class="btn-check"
|
|
||||||
name="heartbeatBarTheme"
|
|
||||||
autocomplete="off"
|
|
||||||
value="bottom"
|
|
||||||
/>
|
|
||||||
<label class="btn btn-outline-primary" for="btncheck5">
|
|
||||||
{{ $t("Bottom") }}
|
|
||||||
</label>
|
|
||||||
|
|
||||||
<input
|
|
||||||
id="btncheck6"
|
|
||||||
v-model="$root.userHeartbeatBar"
|
|
||||||
type="radio"
|
|
||||||
class="btn-check"
|
|
||||||
name="heartbeatBarTheme"
|
|
||||||
autocomplete="off"
|
|
||||||
value="none"
|
|
||||||
/>
|
|
||||||
<label class="btn btn-outline-primary" for="btncheck6">
|
|
||||||
{{ $t("None") }}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Timeline -->
|
|
||||||
<div class="my-4">
|
|
||||||
<label class="form-label">{{ $t("styleElapsedTime") }}</label>
|
|
||||||
<div>
|
|
||||||
<div class="btn-group" role="group">
|
|
||||||
<input
|
|
||||||
id="styleElapsedTimeShowNoLine"
|
|
||||||
v-model="$root.styleElapsedTime"
|
|
||||||
type="radio"
|
|
||||||
class="btn-check"
|
|
||||||
name="styleElapsedTime"
|
|
||||||
autocomplete="off"
|
|
||||||
value="no-line"
|
|
||||||
/>
|
|
||||||
<label class="btn btn-outline-primary" for="styleElapsedTimeShowNoLine">
|
|
||||||
{{ $t("styleElapsedTimeShowNoLine") }}
|
|
||||||
</label>
|
|
||||||
|
|
||||||
<input
|
|
||||||
id="styleElapsedTimeShowWithLine"
|
|
||||||
v-model="$root.styleElapsedTime"
|
|
||||||
type="radio"
|
|
||||||
class="btn-check"
|
|
||||||
name="styleElapsedTime"
|
|
||||||
autocomplete="off"
|
|
||||||
value="with-line"
|
|
||||||
/>
|
|
||||||
<label class="btn btn-outline-primary" for="styleElapsedTimeShowWithLine">
|
|
||||||
{{ $t("styleElapsedTimeShowWithLine") }}
|
|
||||||
</label>
|
|
||||||
|
|
||||||
<input
|
|
||||||
id="styleElapsedTimeNone"
|
|
||||||
v-model="$root.styleElapsedTime"
|
|
||||||
type="radio"
|
|
||||||
class="btn-check"
|
|
||||||
name="styleElapsedTime"
|
|
||||||
autocomplete="off"
|
|
||||||
value="none"
|
|
||||||
/>
|
|
||||||
<label class="btn btn-outline-primary" for="styleElapsedTimeNone">
|
|
||||||
{{ $t("None") }}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
export default {
|
import HorizontalTabHeader from "../HorizontalTabHeader.vue";
|
||||||
|
import AppearanceSettings from "./AppearanceSettings.vue";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
components: {
|
||||||
|
HorizontalTabHeader,
|
||||||
|
AppearanceSettings
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
storageLoc: 0,
|
||||||
|
languagesList: this.$i18n.availableLocales.map(lang => {
|
||||||
|
return {
|
||||||
|
value: lang,
|
||||||
|
label: this.$i18n.messages[lang].languageName,
|
||||||
|
};
|
||||||
|
})
|
||||||
|
};
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
settings() {
|
||||||
|
return this.$parent.$parent.$parent.settings;
|
||||||
|
},
|
||||||
|
saveSettings() {
|
||||||
|
return this.$parent.$parent.$parent.saveSettings;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
updateDefaultLanguage(language) {
|
||||||
|
this.updateDefaultAppearance({
|
||||||
|
...this.settings.defaultAppearance,
|
||||||
|
language
|
||||||
|
});
|
||||||
|
},
|
||||||
|
updateDefaultTheme(theme) {
|
||||||
|
this.updateDefaultAppearance({
|
||||||
|
...this.settings.defaultAppearance,
|
||||||
|
theme: theme
|
||||||
|
});
|
||||||
|
},
|
||||||
|
updateDefaultHeartbeatBarTheme(theme) {
|
||||||
|
this.updateDefaultAppearance({
|
||||||
|
...this.settings.defaultAppearance,
|
||||||
|
heartbeatBarTheme: theme
|
||||||
|
});
|
||||||
|
},
|
||||||
|
updateDefaultStyleElapsedTime(styleElapsedTime) {
|
||||||
|
this.updateDefaultAppearance({
|
||||||
|
...this.settings.defaultAppearance,
|
||||||
|
styleElapsedTime
|
||||||
|
});
|
||||||
|
},
|
||||||
|
updateDefaultAppearance(appearance) {
|
||||||
|
this.$parent.$parent.$parent.settings.defaultAppearance = appearance;
|
||||||
|
this.saveSettings();
|
||||||
|
}
|
||||||
|
}
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import "../../assets/vars.scss";
|
@import "../../assets/vars.scss";
|
||||||
|
|
||||||
.btn-check:active + .btn-outline-primary,
|
|
||||||
.btn-check:checked + .btn-outline-primary,
|
|
||||||
.btn-check:hover + .btn-outline-primary {
|
|
||||||
color: #fff;
|
|
||||||
|
|
||||||
.dark & {
|
|
||||||
color: #000;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.dark {
|
.dark {
|
||||||
.list-group-item {
|
.list-group-item {
|
||||||
background-color: $dark-bg2;
|
background-color: $dark-bg2;
|
||||||
|
|
234
src/components/settings/AppearanceSettings.vue
Normal file
234
src/components/settings/AppearanceSettings.vue
Normal file
|
@ -0,0 +1,234 @@
|
||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<div class="my-4">
|
||||||
|
<label for="language" class="form-label">
|
||||||
|
{{ $t("Language") }}
|
||||||
|
</label>
|
||||||
|
<select
|
||||||
|
id="language"
|
||||||
|
:value="language"
|
||||||
|
class="form-select"
|
||||||
|
@input="$emit('update:language', $event.target.value)"
|
||||||
|
>
|
||||||
|
<option
|
||||||
|
v-for="(lang, i) in languages"
|
||||||
|
:key="`Lang${i}`"
|
||||||
|
:value="lang.value"
|
||||||
|
>
|
||||||
|
{{ lang.label }}
|
||||||
|
</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
<div class="my-4">
|
||||||
|
<label for="timezone" class="form-label">{{ $t("Theme") }}</label>
|
||||||
|
<div>
|
||||||
|
<div
|
||||||
|
class="btn-group"
|
||||||
|
role="group"
|
||||||
|
aria-label="Basic checkbox toggle button group"
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
id="btncheck1"
|
||||||
|
:checked="userTheme == 'light'"
|
||||||
|
type="radio"
|
||||||
|
class="btn-check"
|
||||||
|
name="theme"
|
||||||
|
autocomplete="off"
|
||||||
|
value="light"
|
||||||
|
@input="$emit('update:userTheme', 'light')"
|
||||||
|
/>
|
||||||
|
<label class="btn btn-outline-primary" for="btncheck1">
|
||||||
|
{{ $t("Light") }}
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<input
|
||||||
|
id="btncheck2"
|
||||||
|
:checked="userTheme == 'dark'"
|
||||||
|
type="radio"
|
||||||
|
class="btn-check"
|
||||||
|
name="theme"
|
||||||
|
autocomplete="off"
|
||||||
|
value="dark"
|
||||||
|
@input="$emit('update:userTheme', 'dark')"
|
||||||
|
/>
|
||||||
|
<label class="btn btn-outline-primary" for="btncheck2">
|
||||||
|
{{ $t("Dark") }}
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<input
|
||||||
|
id="btncheck3"
|
||||||
|
:checked="userTheme == 'auto'"
|
||||||
|
type="radio"
|
||||||
|
class="btn-check"
|
||||||
|
name="theme"
|
||||||
|
autocomplete="off"
|
||||||
|
value="auto"
|
||||||
|
@input="$emit('update:userTheme', 'auto')"
|
||||||
|
/>
|
||||||
|
<label class="btn btn-outline-primary" for="btncheck3">
|
||||||
|
{{ $t("Auto") }}
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="my-4">
|
||||||
|
<label class="form-label">{{ $t("Theme - Heartbeat Bar") }}</label>
|
||||||
|
<div>
|
||||||
|
<div
|
||||||
|
class="btn-group"
|
||||||
|
role="group"
|
||||||
|
aria-label="Basic checkbox toggle button group"
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
id="btncheck4"
|
||||||
|
:checked="userHeartbeatBar == 'normal'"
|
||||||
|
type="radio"
|
||||||
|
class="btn-check"
|
||||||
|
name="heartbeatBarTheme"
|
||||||
|
autocomplete="off"
|
||||||
|
value="normal"
|
||||||
|
@input="$emit('update:userHeartbeatBar', 'normal')"
|
||||||
|
/>
|
||||||
|
<label class="btn btn-outline-primary" for="btncheck4">
|
||||||
|
{{ $t("Normal") }}
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<input
|
||||||
|
id="btncheck5"
|
||||||
|
:checked="userHeartbeatBar == 'bottom'"
|
||||||
|
type="radio"
|
||||||
|
class="btn-check"
|
||||||
|
name="heartbeatBarTheme"
|
||||||
|
autocomplete="off"
|
||||||
|
value="bottom"
|
||||||
|
@input="$emit('update:userHeartbeatBar', 'bottom')"
|
||||||
|
/>
|
||||||
|
<label class="btn btn-outline-primary" for="btncheck5">
|
||||||
|
{{ $t("Bottom") }}
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<input
|
||||||
|
id="btncheck6"
|
||||||
|
:checked="userHeartbeatBar == 'none'"
|
||||||
|
type="radio"
|
||||||
|
class="btn-check"
|
||||||
|
name="heartbeatBarTheme"
|
||||||
|
autocomplete="off"
|
||||||
|
value="none"
|
||||||
|
@input="$emit('update:userHeartbeatBar', 'none')"
|
||||||
|
/>
|
||||||
|
<label class="btn btn-outline-primary" for="btncheck6">
|
||||||
|
{{ $t("None") }}
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Timeline -->
|
||||||
|
<div class="my-4">
|
||||||
|
<label class="form-label">{{ $t("styleElapsedTime") }}</label>
|
||||||
|
<div>
|
||||||
|
<div class="btn-group" role="group">
|
||||||
|
<input
|
||||||
|
id="styleElapsedTimeShowNoLine"
|
||||||
|
:checked="styleElapsedTime == 'no-line'"
|
||||||
|
type="radio"
|
||||||
|
class="btn-check"
|
||||||
|
name="styleElapsedTime"
|
||||||
|
autocomplete="off"
|
||||||
|
value="no-line"
|
||||||
|
@input="$emit('update:styleElapsedTime', 'no-line')"
|
||||||
|
/>
|
||||||
|
<label
|
||||||
|
class="btn btn-outline-primary"
|
||||||
|
for="styleElapsedTimeShowNoLine"
|
||||||
|
>
|
||||||
|
{{ $t("styleElapsedTimeShowNoLine") }}
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<input
|
||||||
|
id="styleElapsedTimeShowWithLine"
|
||||||
|
:checked="styleElapsedTime == 'with-line'"
|
||||||
|
type="radio"
|
||||||
|
class="btn-check"
|
||||||
|
name="styleElapsedTime"
|
||||||
|
autocomplete="off"
|
||||||
|
value="with-line"
|
||||||
|
@input="$emit('update:styleElapsedTime', 'with-line')"
|
||||||
|
/>
|
||||||
|
<label
|
||||||
|
class="btn btn-outline-primary"
|
||||||
|
for="styleElapsedTimeShowWithLine"
|
||||||
|
>
|
||||||
|
{{ $t("styleElapsedTimeShowWithLine") }}
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<input
|
||||||
|
id="styleElapsedTimeNone"
|
||||||
|
:checked="styleElapsedTime == 'none'"
|
||||||
|
type="radio"
|
||||||
|
class="btn-check"
|
||||||
|
name="styleElapsedTime"
|
||||||
|
autocomplete="off"
|
||||||
|
value="none"
|
||||||
|
@input="$emit('update:styleElapsedTime', 'none')"
|
||||||
|
/>
|
||||||
|
<label
|
||||||
|
class="btn btn-outline-primary"
|
||||||
|
for="styleElapsedTimeNone"
|
||||||
|
>
|
||||||
|
{{ $t("None") }}
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="js">
|
||||||
|
export default {
|
||||||
|
props: {
|
||||||
|
languages: {
|
||||||
|
type: Array,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
language: {
|
||||||
|
type: String,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
userTheme: {
|
||||||
|
type: String,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
userHeartbeatBar: {
|
||||||
|
type: String,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
styleElapsedTime: {
|
||||||
|
type: String,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
emits: [
|
||||||
|
"update:language",
|
||||||
|
"update:userTheme",
|
||||||
|
"update:userHeartbeatBar",
|
||||||
|
"update:styleElapsedTime",
|
||||||
|
],
|
||||||
|
data() {
|
||||||
|
return {};
|
||||||
|
},
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.btn-check:active + .btn-outline-primary,
|
||||||
|
.btn-check:checked + .btn-outline-primary,
|
||||||
|
.btn-check:hover + .btn-outline-primary {
|
||||||
|
color: #fff;
|
||||||
|
|
||||||
|
.dark & {
|
||||||
|
color: #000;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -13,6 +13,24 @@ export default {
|
||||||
},
|
},
|
||||||
|
|
||||||
mounted() {
|
mounted() {
|
||||||
|
if (window.defaultAppearance) {
|
||||||
|
if (window.defaultAppearance.language) {
|
||||||
|
localStorage.locale = window.defaultAppearance.language;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (window.defaultAppearance.theme) {
|
||||||
|
this.userTheme = window.defaultAppearance.theme;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (window.defaultAppearance.heartbeatBarTheme) {
|
||||||
|
this.userHeartbeatBar = window.defaultAppearance.heartbeatBarTheme;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (window.defaultAppearance.styleElapsedTime) {
|
||||||
|
this.styleElapsedTime = window.defaultAppearance.styleElapsedTime;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Default Light
|
// Default Light
|
||||||
if (! this.userTheme) {
|
if (! this.userTheme) {
|
||||||
this.userTheme = "auto";
|
this.userTheme = "auto";
|
||||||
|
|
|
@ -183,6 +183,15 @@ export default {
|
||||||
this.settings.trustProxy = false;
|
this.settings.trustProxy = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (this.settings.defaultAppearance === undefined) {
|
||||||
|
this.settings.defaultAppearance = {
|
||||||
|
theme: null,
|
||||||
|
language: null,
|
||||||
|
heartbeatBarTheme: null,
|
||||||
|
styleElapsedTime: null,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
this.settingsLoaded = true;
|
this.settingsLoaded = true;
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
Loading…
Add table
Reference in a new issue