Merge branch 'master' into import-export
This commit is contained in:
commit
242e494cb5
|
@ -27,6 +27,7 @@ VPS is sponsored by Uptime Kuma sponsors on [Open Collective](https://opencollec
|
||||||
* Fancy, Reactive, Fast UI/UX.
|
* Fancy, Reactive, Fast UI/UX.
|
||||||
* Notifications via Telegram, Discord, Gotify, Slack, Pushover, Email (SMTP), and [70+ notification services, click here for the full list](https://github.com/louislam/uptime-kuma/issues/284).
|
* Notifications via Telegram, Discord, Gotify, Slack, Pushover, Email (SMTP), and [70+ notification services, click here for the full list](https://github.com/louislam/uptime-kuma/issues/284).
|
||||||
* 20 seconds interval.
|
* 20 seconds interval.
|
||||||
|
* [Multi Languages](https://github.com/louislam/uptime-kuma/tree/master/src/languages)
|
||||||
|
|
||||||
## 🔧 How to Install
|
## 🔧 How to Install
|
||||||
|
|
||||||
|
|
23
dockerfile
23
dockerfile
|
@ -1,25 +1,30 @@
|
||||||
# DON'T UPDATE TO alpine3.13, 1.14, see #41.
|
FROM node:14-bullseye-slim AS release
|
||||||
FROM node:14-alpine3.12 AS release
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
|
# install dependencies
|
||||||
|
RUN apt update && apt --yes install python3 python3-pip python3-dev git g++ make iputils-ping
|
||||||
|
RUN ln -s /usr/bin/python3 /usr/bin/python
|
||||||
|
|
||||||
# split the sqlite install here, so that it can caches the arm prebuilt
|
# split the sqlite install here, so that it can caches the arm prebuilt
|
||||||
RUN apk add --no-cache --virtual .build-deps make g++ python3 python3-dev git && \
|
RUN npm install mapbox/node-sqlite3#593c9d
|
||||||
ln -s /usr/bin/python3 /usr/bin/python && \
|
|
||||||
npm install mapbox/node-sqlite3#593c9d && \
|
|
||||||
apk del .build-deps && \
|
|
||||||
rm -f /usr/bin/python
|
|
||||||
|
|
||||||
# Install apprise
|
# Install apprise
|
||||||
RUN apk add --no-cache python3 py3-cryptography py3-pip py3-six py3-yaml py3-click py3-markdown py3-requests py3-requests-oauthlib
|
RUN apt --yes install python3-cryptography python3-six python3-yaml python3-click python3-markdown python3-requests python3-requests-oauthlib
|
||||||
RUN pip3 --no-cache-dir install apprise && \
|
RUN pip3 --no-cache-dir install apprise && \
|
||||||
rm -rf /root/.cache
|
rm -rf /root/.cache
|
||||||
|
|
||||||
|
# additional package should be added here, since we don't want to re-compile the arm prebuilt again
|
||||||
|
|
||||||
|
# add sqlite3 cli for debugging in the future
|
||||||
|
RUN apt --yes install sqlite3
|
||||||
|
|
||||||
|
|
||||||
COPY . .
|
COPY . .
|
||||||
RUN npm install --legacy-peer-deps && npm run build && npm prune
|
RUN npm install --legacy-peer-deps && npm run build && npm prune
|
||||||
|
|
||||||
EXPOSE 3001
|
EXPOSE 3001
|
||||||
VOLUME ["/app/data"]
|
VOLUME ["/app/data"]
|
||||||
HEALTHCHECK --interval=60s --timeout=30s --start-period=300s CMD node extra/healthcheck.js
|
HEALTHCHECK --interval=600s --timeout=130s --start-period=300s CMD node extra/healthcheck.js
|
||||||
CMD ["node", "server/server.js"]
|
CMD ["node", "server/server.js"]
|
||||||
|
|
||||||
FROM release AS nightly
|
FROM release AS nightly
|
||||||
|
|
|
@ -0,0 +1,26 @@
|
||||||
|
# DON'T UPDATE TO alpine3.13, 1.14, see #41.
|
||||||
|
FROM node:14-alpine3.12 AS release
|
||||||
|
WORKDIR /app
|
||||||
|
|
||||||
|
# split the sqlite install here, so that it can caches the arm prebuilt
|
||||||
|
RUN apk add --no-cache --virtual .build-deps make g++ python3 python3-dev git && \
|
||||||
|
ln -s /usr/bin/python3 /usr/bin/python && \
|
||||||
|
npm install mapbox/node-sqlite3#593c9d && \
|
||||||
|
apk del .build-deps && \
|
||||||
|
rm -f /usr/bin/python
|
||||||
|
|
||||||
|
# Install apprise
|
||||||
|
RUN apk add --no-cache python3 py3-cryptography py3-pip py3-six py3-yaml py3-click py3-markdown py3-requests py3-requests-oauthlib
|
||||||
|
RUN pip3 --no-cache-dir install apprise && \
|
||||||
|
rm -rf /root/.cache
|
||||||
|
|
||||||
|
COPY . .
|
||||||
|
RUN npm install --legacy-peer-deps && npm run build && npm prune
|
||||||
|
|
||||||
|
EXPOSE 3001
|
||||||
|
VOLUME ["/app/data"]
|
||||||
|
HEALTHCHECK --interval=600s --timeout=130s --start-period=300s CMD node extra/healthcheck.js
|
||||||
|
CMD ["node", "server/server.js"]
|
||||||
|
|
||||||
|
FROM release AS nightly
|
||||||
|
RUN npm run mark-as-nightly
|
|
@ -1,19 +1,31 @@
|
||||||
let http = require("http");
|
process.env.NODE_TLS_REJECT_UNAUTHORIZED = "0";
|
||||||
|
|
||||||
|
let client;
|
||||||
|
|
||||||
|
if (process.env.SSL_KEY && process.env.SSL_CERT) {
|
||||||
|
client = require("https");
|
||||||
|
} else {
|
||||||
|
client = require("http");
|
||||||
|
}
|
||||||
|
|
||||||
let options = {
|
let options = {
|
||||||
host: "localhost",
|
host: process.env.HOST || "127.0.0.1",
|
||||||
port: "3001",
|
port: parseInt(process.env.PORT) || 3001,
|
||||||
timeout: 2000,
|
timeout: 120 * 1000,
|
||||||
};
|
};
|
||||||
let request = http.request(options, (res) => {
|
|
||||||
console.log(`STATUS: ${res.statusCode}`);
|
let request = client.request(options, (res) => {
|
||||||
if (res.statusCode == 200) {
|
console.log(`Health Check OK [Res Code: ${res.statusCode}]`);
|
||||||
|
if (res.statusCode === 200) {
|
||||||
process.exit(0);
|
process.exit(0);
|
||||||
} else {
|
} else {
|
||||||
process.exit(1);
|
process.exit(1);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
request.on("error", function (err) {
|
request.on("error", function (err) {
|
||||||
console.log("ERROR");
|
console.error("Health Check ERROR");
|
||||||
process.exit(1);
|
process.exit(1);
|
||||||
});
|
});
|
||||||
|
|
||||||
request.end();
|
request.end();
|
||||||
|
|
|
@ -23,6 +23,8 @@ if (! exists) {
|
||||||
pkg.version = newVersion;
|
pkg.version = newVersion;
|
||||||
pkg.scripts.setup = pkg.scripts.setup.replaceAll(oldVersion, newVersion);
|
pkg.scripts.setup = pkg.scripts.setup.replaceAll(oldVersion, newVersion);
|
||||||
pkg.scripts["build-docker"] = pkg.scripts["build-docker"].replaceAll(oldVersion, newVersion);
|
pkg.scripts["build-docker"] = pkg.scripts["build-docker"].replaceAll(oldVersion, newVersion);
|
||||||
|
pkg.scripts["build-docker-alpine"] = pkg.scripts["build-docker-alpine"].replaceAll(oldVersion, newVersion);
|
||||||
|
pkg.scripts["build-docker-debian"] = pkg.scripts["build-docker-debian"].replaceAll(oldVersion, newVersion);
|
||||||
fs.writeFileSync("package.json", JSON.stringify(pkg, null, 4) + "\n");
|
fs.writeFileSync("package.json", JSON.stringify(pkg, null, 4) + "\n");
|
||||||
|
|
||||||
commit(newVersion);
|
commit(newVersion);
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
11
package.json
11
package.json
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"name": "uptime-kuma",
|
"name": "uptime-kuma",
|
||||||
"version": "1.5.2",
|
"version": "1.5.3",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
|
@ -18,11 +18,12 @@
|
||||||
"start-server": "node server/server.js",
|
"start-server": "node server/server.js",
|
||||||
"build": "vite build",
|
"build": "vite build",
|
||||||
"vite-preview-dist": "vite preview --host",
|
"vite-preview-dist": "vite preview --host",
|
||||||
"build-docker": "docker buildx build --platform linux/amd64,linux/arm64,linux/arm/v7 -t louislam/uptime-kuma -t louislam/uptime-kuma:1 -t louislam/uptime-kuma:1.5.2 --target release . --push",
|
"build-docker": "npm run build-docker-alpine && npm run build-docker-debian",
|
||||||
|
"build-docker-alpine": "docker buildx build -f dockerfile-alpine --platform linux/amd64,linux/arm64,linux/arm/v7 -t louislam/uptime-kuma:alpine -t louislam/uptime-kuma:1-alpine -t louislam/uptime-kuma:1.5.3-alpine --target release . --push",
|
||||||
|
"build-docker-debian": "docker buildx build --platform linux/amd64,linux/arm64,linux/arm/v7 -t louislam/uptime-kuma -t louislam/uptime-kuma:1 -t louislam/uptime-kuma:1.5.3 -t louislam/uptime-kuma:debian -t louislam/uptime-kuma:1-debian -t louislam/uptime-kuma:1.5.3-debian --target release . --push",
|
||||||
"build-docker-nightly": "docker buildx build --platform linux/amd64,linux/arm64,linux/arm/v7 -t louislam/uptime-kuma:nightly --target nightly . --push",
|
"build-docker-nightly": "docker buildx build --platform linux/amd64,linux/arm64,linux/arm/v7 -t louislam/uptime-kuma:nightly --target nightly . --push",
|
||||||
"build-docker-nightly-amd64": "docker buildx build --platform linux/amd64 -t louislam/uptime-kuma:nightly-amd64 --target nightly . --push --progress plain",
|
"build-docker-nightly-amd64": "docker buildx build --platform linux/amd64 -t louislam/uptime-kuma:nightly-amd64 --target nightly . --push --progress plain",
|
||||||
"build-docker-1.5.0-debian": "docker buildx build --platform linux/amd64,linux/arm64,linux/arm/v7 -t louislam/uptime-kuma:1.5.0-debian --target release . --push",
|
"setup": "git checkout 1.5.3 && npm install --legacy-peer-deps && node node_modules/esbuild/install.js && npm run build && npm prune",
|
||||||
"setup": "git checkout 1.5.2 && npm install --legacy-peer-deps && node node_modules/esbuild/install.js && npm run build && npm prune",
|
|
||||||
"update-version": "node extra/update-version.js",
|
"update-version": "node extra/update-version.js",
|
||||||
"mark-as-nightly": "node extra/mark-as-nightly.js",
|
"mark-as-nightly": "node extra/mark-as-nightly.js",
|
||||||
"reset-password": "node extra/reset-password.js",
|
"reset-password": "node extra/reset-password.js",
|
||||||
|
@ -64,7 +65,7 @@
|
||||||
"sqlite3": "github:mapbox/node-sqlite3#593c9d",
|
"sqlite3": "github:mapbox/node-sqlite3#593c9d",
|
||||||
"tcp-ping": "^0.1.1",
|
"tcp-ping": "^0.1.1",
|
||||||
"v-pagination-3": "^0.1.6",
|
"v-pagination-3": "^0.1.6",
|
||||||
"vue": "^3.2.6",
|
"vue": "^3.2.8",
|
||||||
"vue-chart-3": "^0.5.7",
|
"vue-chart-3": "^0.5.7",
|
||||||
"vue-confirm-dialog": "^1.0.2",
|
"vue-confirm-dialog": "^1.0.2",
|
||||||
"vue-i18n": "^9.1.7",
|
"vue-i18n": "^9.1.7",
|
||||||
|
|
|
@ -0,0 +1,91 @@
|
||||||
|
/*
|
||||||
|
* For Client Socket
|
||||||
|
*/
|
||||||
|
const { TimeLogger } = require("../src/util");
|
||||||
|
const { R } = require("redbean-node");
|
||||||
|
const { io } = require("./server");
|
||||||
|
|
||||||
|
async function sendNotificationList(socket) {
|
||||||
|
const timeLogger = new TimeLogger();
|
||||||
|
|
||||||
|
let result = [];
|
||||||
|
let list = await R.find("notification", " user_id = ? ", [
|
||||||
|
socket.userID,
|
||||||
|
]);
|
||||||
|
|
||||||
|
for (let bean of list) {
|
||||||
|
result.push(bean.export())
|
||||||
|
}
|
||||||
|
|
||||||
|
io.to(socket.userID).emit("notificationList", result)
|
||||||
|
|
||||||
|
timeLogger.print("Send Notification List");
|
||||||
|
|
||||||
|
return list;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Send Heartbeat History list to socket
|
||||||
|
* @param toUser True = send to all browsers with the same user id, False = send to the current browser only
|
||||||
|
* @param overwrite Overwrite client-side's heartbeat list
|
||||||
|
*/
|
||||||
|
async function sendHeartbeatList(socket, monitorID, toUser = false, overwrite = false) {
|
||||||
|
const timeLogger = new TimeLogger();
|
||||||
|
|
||||||
|
let list = await R.find("heartbeat", `
|
||||||
|
monitor_id = ?
|
||||||
|
ORDER BY time DESC
|
||||||
|
LIMIT 100
|
||||||
|
`, [
|
||||||
|
monitorID,
|
||||||
|
])
|
||||||
|
|
||||||
|
let result = [];
|
||||||
|
|
||||||
|
for (let bean of list) {
|
||||||
|
result.unshift(bean.toJSON());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (toUser) {
|
||||||
|
io.to(socket.userID).emit("heartbeatList", monitorID, result, overwrite);
|
||||||
|
} else {
|
||||||
|
socket.emit("heartbeatList", monitorID, result, overwrite);
|
||||||
|
}
|
||||||
|
|
||||||
|
timeLogger.print(`[Monitor: ${monitorID}] sendHeartbeatList`);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Important Heart beat list (aka event list)
|
||||||
|
* @param socket
|
||||||
|
* @param monitorID
|
||||||
|
* @param toUser True = send to all browsers with the same user id, False = send to the current browser only
|
||||||
|
* @param overwrite Overwrite client-side's heartbeat list
|
||||||
|
*/
|
||||||
|
async function sendImportantHeartbeatList(socket, monitorID, toUser = false, overwrite = false) {
|
||||||
|
const timeLogger = new TimeLogger();
|
||||||
|
|
||||||
|
let list = await R.find("heartbeat", `
|
||||||
|
monitor_id = ?
|
||||||
|
AND important = 1
|
||||||
|
ORDER BY time DESC
|
||||||
|
LIMIT 500
|
||||||
|
`, [
|
||||||
|
monitorID,
|
||||||
|
])
|
||||||
|
|
||||||
|
timeLogger.print(`[Monitor: ${monitorID}] sendImportantHeartbeatList`);
|
||||||
|
|
||||||
|
if (toUser) {
|
||||||
|
io.to(socket.userID).emit("importantHeartbeatList", monitorID, list, overwrite);
|
||||||
|
} else {
|
||||||
|
socket.emit("importantHeartbeatList", monitorID, list, overwrite);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
sendNotificationList,
|
||||||
|
sendImportantHeartbeatList,
|
||||||
|
sendHeartbeatList,
|
||||||
|
}
|
|
@ -4,6 +4,8 @@ const FormData = require("form-data");
|
||||||
const nodemailer = require("nodemailer");
|
const nodemailer = require("nodemailer");
|
||||||
const child_process = require("child_process");
|
const child_process = require("child_process");
|
||||||
|
|
||||||
|
const { UP, DOWN } = require("../src/util");
|
||||||
|
|
||||||
class Notification {
|
class Notification {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -80,7 +82,7 @@ class Notification {
|
||||||
}
|
}
|
||||||
|
|
||||||
} else if (notification.type === "smtp") {
|
} else if (notification.type === "smtp") {
|
||||||
return await Notification.smtp(notification, msg)
|
return await Notification.smtp(notification, msg, heartbeatJSON)
|
||||||
|
|
||||||
} else if (notification.type === "discord") {
|
} else if (notification.type === "discord") {
|
||||||
try {
|
try {
|
||||||
|
@ -109,7 +111,7 @@ class Notification {
|
||||||
}
|
}
|
||||||
|
|
||||||
// If heartbeatJSON is not null, we go into the normal alerting loop.
|
// If heartbeatJSON is not null, we go into the normal alerting loop.
|
||||||
if (heartbeatJSON["status"] == 0) {
|
if (heartbeatJSON["status"] == DOWN) {
|
||||||
let discorddowndata = {
|
let discorddowndata = {
|
||||||
username: discordDisplayName,
|
username: discordDisplayName,
|
||||||
embeds: [{
|
embeds: [{
|
||||||
|
@ -139,7 +141,7 @@ class Notification {
|
||||||
await axios.post(notification.discordWebhookUrl, discorddowndata)
|
await axios.post(notification.discordWebhookUrl, discorddowndata)
|
||||||
return okMsg;
|
return okMsg;
|
||||||
|
|
||||||
} else if (heartbeatJSON["status"] == 1) {
|
} else if (heartbeatJSON["status"] == UP) {
|
||||||
let discordupdata = {
|
let discordupdata = {
|
||||||
username: discordDisplayName,
|
username: discordDisplayName,
|
||||||
embeds: [{
|
embeds: [{
|
||||||
|
@ -343,7 +345,7 @@ class Notification {
|
||||||
const mattermostIconEmoji = notification.mattermosticonemo;
|
const mattermostIconEmoji = notification.mattermosticonemo;
|
||||||
const mattermostIconUrl = notification.mattermosticonurl;
|
const mattermostIconUrl = notification.mattermosticonurl;
|
||||||
|
|
||||||
if (heartbeatJSON["status"] == 0) {
|
if (heartbeatJSON["status"] == DOWN) {
|
||||||
let mattermostdowndata = {
|
let mattermostdowndata = {
|
||||||
username: mattermostUserName,
|
username: mattermostUserName,
|
||||||
text: "Uptime Kuma Alert",
|
text: "Uptime Kuma Alert",
|
||||||
|
@ -387,7 +389,7 @@ class Notification {
|
||||||
mattermostdowndata
|
mattermostdowndata
|
||||||
);
|
);
|
||||||
return okMsg;
|
return okMsg;
|
||||||
} else if (heartbeatJSON["status"] == 1) {
|
} else if (heartbeatJSON["status"] == UP) {
|
||||||
let mattermostupdata = {
|
let mattermostupdata = {
|
||||||
username: mattermostUserName,
|
username: mattermostUserName,
|
||||||
text: "Uptime Kuma Alert",
|
text: "Uptime Kuma Alert",
|
||||||
|
@ -489,7 +491,7 @@ class Notification {
|
||||||
return okMsg;
|
return okMsg;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (heartbeatJSON["status"] == 0) {
|
if (heartbeatJSON["status"] == DOWN) {
|
||||||
let downdata = {
|
let downdata = {
|
||||||
"title": "UptimeKuma Alert: " + monitorJSON["name"],
|
"title": "UptimeKuma Alert: " + monitorJSON["name"],
|
||||||
"body": "[🔴 Down] " + heartbeatJSON["msg"] + "\nTime (UTC): " + heartbeatJSON["time"],
|
"body": "[🔴 Down] " + heartbeatJSON["msg"] + "\nTime (UTC): " + heartbeatJSON["time"],
|
||||||
|
@ -498,7 +500,7 @@ class Notification {
|
||||||
return okMsg;
|
return okMsg;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (heartbeatJSON["status"] == 1) {
|
if (heartbeatJSON["status"] == UP) {
|
||||||
let updata = {
|
let updata = {
|
||||||
"title": "UptimeKuma Alert: " + monitorJSON["name"],
|
"title": "UptimeKuma Alert: " + monitorJSON["name"],
|
||||||
"body": "[✅ Up] " + heartbeatJSON["msg"] + "\nTime (UTC): " + heartbeatJSON["time"],
|
"body": "[✅ Up] " + heartbeatJSON["msg"] + "\nTime (UTC): " + heartbeatJSON["time"],
|
||||||
|
@ -527,14 +529,14 @@ class Notification {
|
||||||
"body": "Testing Successful.",
|
"body": "Testing Successful.",
|
||||||
}
|
}
|
||||||
await axios.post(pushbulletUrl, testdata, config)
|
await axios.post(pushbulletUrl, testdata, config)
|
||||||
} else if (heartbeatJSON["status"] == 0) {
|
} else if (heartbeatJSON["status"] == DOWN) {
|
||||||
let downdata = {
|
let downdata = {
|
||||||
"type": "note",
|
"type": "note",
|
||||||
"title": "UptimeKuma Alert: " + monitorJSON["name"],
|
"title": "UptimeKuma Alert: " + monitorJSON["name"],
|
||||||
"body": "[🔴 Down] " + heartbeatJSON["msg"] + "\nTime (UTC): " + heartbeatJSON["time"],
|
"body": "[🔴 Down] " + heartbeatJSON["msg"] + "\nTime (UTC): " + heartbeatJSON["time"],
|
||||||
}
|
}
|
||||||
await axios.post(pushbulletUrl, downdata, config)
|
await axios.post(pushbulletUrl, downdata, config)
|
||||||
} else if (heartbeatJSON["status"] == 1) {
|
} else if (heartbeatJSON["status"] == UP) {
|
||||||
let updata = {
|
let updata = {
|
||||||
"type": "note",
|
"type": "note",
|
||||||
"title": "UptimeKuma Alert: " + monitorJSON["name"],
|
"title": "UptimeKuma Alert: " + monitorJSON["name"],
|
||||||
|
@ -566,7 +568,7 @@ class Notification {
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
await axios.post(lineAPIUrl, testMessage, config)
|
await axios.post(lineAPIUrl, testMessage, config)
|
||||||
} else if (heartbeatJSON["status"] == 0) {
|
} else if (heartbeatJSON["status"] == DOWN) {
|
||||||
let downMessage = {
|
let downMessage = {
|
||||||
"to": notification.lineUserID,
|
"to": notification.lineUserID,
|
||||||
"messages": [
|
"messages": [
|
||||||
|
@ -577,7 +579,7 @@ class Notification {
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
await axios.post(lineAPIUrl, downMessage, config)
|
await axios.post(lineAPIUrl, downMessage, config)
|
||||||
} else if (heartbeatJSON["status"] == 1) {
|
} else if (heartbeatJSON["status"] == UP) {
|
||||||
let upMessage = {
|
let upMessage = {
|
||||||
"to": notification.lineUserID,
|
"to": notification.lineUserID,
|
||||||
"messages": [
|
"messages": [
|
||||||
|
@ -634,7 +636,7 @@ class Notification {
|
||||||
await R.trash(bean)
|
await R.trash(bean)
|
||||||
}
|
}
|
||||||
|
|
||||||
static async smtp(notification, msg) {
|
static async smtp(notification, msg, heartbeatJSON = null) {
|
||||||
|
|
||||||
const config = {
|
const config = {
|
||||||
host: notification.smtpHost,
|
host: notification.smtpHost,
|
||||||
|
@ -652,12 +654,17 @@ class Notification {
|
||||||
|
|
||||||
let transporter = nodemailer.createTransport(config);
|
let transporter = nodemailer.createTransport(config);
|
||||||
|
|
||||||
|
let bodyTextContent = msg;
|
||||||
|
if(heartbeatJSON) {
|
||||||
|
bodyTextContent = `${msg}\nTime (UTC): ${heartbeatJSON["time"]}`;
|
||||||
|
}
|
||||||
|
|
||||||
// send mail with defined transport object
|
// send mail with defined transport object
|
||||||
await transporter.sendMail({
|
await transporter.sendMail({
|
||||||
from: `"Uptime Kuma" <${notification.smtpFrom}>`,
|
from: `"Uptime Kuma" <${notification.smtpFrom}>`,
|
||||||
to: notification.smtpTo,
|
to: notification.smtpTo,
|
||||||
subject: msg,
|
subject: msg,
|
||||||
text: msg,
|
text: bodyTextContent,
|
||||||
});
|
});
|
||||||
|
|
||||||
return "Sent Successfully.";
|
return "Sent Successfully.";
|
||||||
|
|
138
server/server.js
138
server/server.js
|
@ -82,7 +82,12 @@ if (sslKey && sslCert) {
|
||||||
}
|
}
|
||||||
|
|
||||||
const io = new Server(server);
|
const io = new Server(server);
|
||||||
app.use(express.json())
|
module.exports.io = io;
|
||||||
|
|
||||||
|
// Must be after io instantiation
|
||||||
|
const { sendNotificationList, sendHeartbeatList, sendImportantHeartbeatList } = require("./client");
|
||||||
|
|
||||||
|
app.use(express.json());
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Total WebSocket client connected to server currently, no actual use
|
* Total WebSocket client connected to server currently, no actual use
|
||||||
|
@ -657,6 +662,76 @@ let indexHTML = fs.readFileSync("./dist/index.html").toString();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
socket.on("clearEvents", async (monitorID, callback) => {
|
||||||
|
try {
|
||||||
|
checkLogin(socket)
|
||||||
|
|
||||||
|
console.log(`Clear Events Monitor: ${monitorID} User ID: ${socket.userID}`)
|
||||||
|
|
||||||
|
await R.exec("UPDATE heartbeat SET msg = ?, important = ? WHERE monitor_id = ? ", [
|
||||||
|
"",
|
||||||
|
"0",
|
||||||
|
monitorID,
|
||||||
|
]);
|
||||||
|
|
||||||
|
await sendImportantHeartbeatList(socket, monitorID, true, true);
|
||||||
|
|
||||||
|
callback({
|
||||||
|
ok: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
} catch (e) {
|
||||||
|
callback({
|
||||||
|
ok: false,
|
||||||
|
msg: e.message,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
socket.on("clearHeartbeats", async (monitorID, callback) => {
|
||||||
|
try {
|
||||||
|
checkLogin(socket)
|
||||||
|
|
||||||
|
console.log(`Clear Heartbeats Monitor: ${monitorID} User ID: ${socket.userID}`)
|
||||||
|
|
||||||
|
await R.exec("DELETE FROM heartbeat WHERE monitor_id = ?", [
|
||||||
|
monitorID
|
||||||
|
]);
|
||||||
|
|
||||||
|
await sendHeartbeatList(socket, monitorID, true, true);
|
||||||
|
|
||||||
|
callback({
|
||||||
|
ok: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
} catch (e) {
|
||||||
|
callback({
|
||||||
|
ok: false,
|
||||||
|
msg: e.message,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
socket.on("clearStatistics", async (callback) => {
|
||||||
|
try {
|
||||||
|
checkLogin(socket)
|
||||||
|
|
||||||
|
console.log(`Clear Statistics User ID: ${socket.userID}`)
|
||||||
|
|
||||||
|
await R.exec("DELETE FROM heartbeat");
|
||||||
|
|
||||||
|
callback({
|
||||||
|
ok: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
} catch (e) {
|
||||||
|
callback({
|
||||||
|
ok: false,
|
||||||
|
msg: e.message,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
debug("added all socket handlers")
|
debug("added all socket handlers")
|
||||||
|
|
||||||
// ***************************
|
// ***************************
|
||||||
|
@ -725,25 +800,6 @@ async function sendMonitorList(socket) {
|
||||||
return list;
|
return list;
|
||||||
}
|
}
|
||||||
|
|
||||||
async function sendNotificationList(socket) {
|
|
||||||
const timeLogger = new TimeLogger();
|
|
||||||
|
|
||||||
let result = [];
|
|
||||||
let list = await R.find("notification", " user_id = ? ", [
|
|
||||||
socket.userID,
|
|
||||||
]);
|
|
||||||
|
|
||||||
for (let bean of list) {
|
|
||||||
result.push(bean.export())
|
|
||||||
}
|
|
||||||
|
|
||||||
io.to(socket.userID).emit("notificationList", result)
|
|
||||||
|
|
||||||
timeLogger.print("Send Notification List");
|
|
||||||
|
|
||||||
return list;
|
|
||||||
}
|
|
||||||
|
|
||||||
async function afterLogin(socket, user) {
|
async function afterLogin(socket, user) {
|
||||||
socket.userID = user.id;
|
socket.userID = user.id;
|
||||||
socket.join(user.id)
|
socket.join(user.id)
|
||||||
|
@ -878,48 +934,6 @@ async function startMonitors() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Send Heartbeat History list to socket
|
|
||||||
*/
|
|
||||||
async function sendHeartbeatList(socket, monitorID) {
|
|
||||||
const timeLogger = new TimeLogger();
|
|
||||||
|
|
||||||
let list = await R.find("heartbeat", `
|
|
||||||
monitor_id = ?
|
|
||||||
ORDER BY time DESC
|
|
||||||
LIMIT 100
|
|
||||||
`, [
|
|
||||||
monitorID,
|
|
||||||
])
|
|
||||||
|
|
||||||
let result = [];
|
|
||||||
|
|
||||||
for (let bean of list) {
|
|
||||||
result.unshift(bean.toJSON())
|
|
||||||
}
|
|
||||||
|
|
||||||
socket.emit("heartbeatList", monitorID, result)
|
|
||||||
|
|
||||||
timeLogger.print(`[Monitor: ${monitorID}] sendHeartbeatList`)
|
|
||||||
}
|
|
||||||
|
|
||||||
async function sendImportantHeartbeatList(socket, monitorID) {
|
|
||||||
const timeLogger = new TimeLogger();
|
|
||||||
|
|
||||||
let list = await R.find("heartbeat", `
|
|
||||||
monitor_id = ?
|
|
||||||
AND important = 1
|
|
||||||
ORDER BY time DESC
|
|
||||||
LIMIT 500
|
|
||||||
`, [
|
|
||||||
monitorID,
|
|
||||||
])
|
|
||||||
|
|
||||||
timeLogger.print(`[Monitor: ${monitorID}] sendImportantHeartbeatList`);
|
|
||||||
|
|
||||||
socket.emit("importantHeartbeatList", monitorID, list)
|
|
||||||
}
|
|
||||||
|
|
||||||
async function shutdownFunction(signal) {
|
async function shutdownFunction(signal) {
|
||||||
console.log("Shutdown requested");
|
console.log("Shutdown requested");
|
||||||
console.log("Called signal: " + signal);
|
console.log("Called signal: " + signal);
|
||||||
|
|
|
@ -16,3 +16,5 @@ $dark-border-color: #1d2634;
|
||||||
$easing-in: cubic-bezier(0.54, 0.78, 0.55, 0.97);
|
$easing-in: cubic-bezier(0.54, 0.78, 0.55, 0.97);
|
||||||
$easing-out: cubic-bezier(0.25, 0.46, 0.45, 0.94);
|
$easing-out: cubic-bezier(0.25, 0.46, 0.45, 0.94);
|
||||||
$easing-in-out: cubic-bezier(0.79, 0.14, 0.15, 0.86);
|
$easing-in-out: cubic-bezier(0.79, 0.14, 0.15, 0.86);
|
||||||
|
|
||||||
|
$dropdown-border-radius: 0.5rem;
|
||||||
|
|
|
@ -31,7 +31,7 @@ export default {
|
||||||
beatWidth: 10,
|
beatWidth: 10,
|
||||||
beatHeight: 30,
|
beatHeight: 30,
|
||||||
hoverScale: 1.5,
|
hoverScale: 1.5,
|
||||||
beatMargin: 3, // Odd number only, even = blurry
|
beatMargin: 4,
|
||||||
move: false,
|
move: false,
|
||||||
maxBeat: -1,
|
maxBeat: -1,
|
||||||
}
|
}
|
||||||
|
@ -122,11 +122,26 @@ export default {
|
||||||
this.$root.heartbeatList[this.monitorId] = [];
|
this.$root.heartbeatList[this.monitorId] = [];
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
mounted() {
|
mounted() {
|
||||||
if (this.size === "small") {
|
if (this.size === "small") {
|
||||||
this.beatWidth = 5.6;
|
this.beatWidth = 5;
|
||||||
this.beatMargin = 2.4;
|
this.beatHeight = 16;
|
||||||
this.beatHeight = 16
|
this.beatMargin = 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Suddenly, have an idea how to handle it universally.
|
||||||
|
// If the pixel * ratio != Integer, then it causes render issue, round it to solve it!!
|
||||||
|
const actualWidth = this.beatWidth * window.devicePixelRatio;
|
||||||
|
const actualMargin = this.beatMargin * window.devicePixelRatio;
|
||||||
|
|
||||||
|
if (! Number.isInteger(actualWidth)) {
|
||||||
|
this.beatWidth = Math.round(actualWidth) / window.devicePixelRatio;
|
||||||
|
console.log(this.beatWidth);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! Number.isInteger(actualMargin)) {
|
||||||
|
this.beatMargin = Math.round(actualMargin) / window.devicePixelRatio;
|
||||||
}
|
}
|
||||||
|
|
||||||
window.addEventListener("resize", this.resize);
|
window.addEventListener("resize", this.resize);
|
||||||
|
|
|
@ -0,0 +1,102 @@
|
||||||
|
<template>
|
||||||
|
<div class="input-group mb-3">
|
||||||
|
<!--
|
||||||
|
Hack - Disable Chrome save password
|
||||||
|
readonly + onfocus
|
||||||
|
https://stackoverflow.com/questions/41217019/how-to-prevent-a-browser-from-storing-passwords
|
||||||
|
-->
|
||||||
|
<input
|
||||||
|
v-model="model"
|
||||||
|
:type="visibility"
|
||||||
|
class="form-control"
|
||||||
|
:placeholder="placeholder"
|
||||||
|
:maxlength="maxlength"
|
||||||
|
:autocomplete="autocomplete"
|
||||||
|
:required="required"
|
||||||
|
:readonly="isReadOnly"
|
||||||
|
@focus="removeReadOnly"
|
||||||
|
>
|
||||||
|
|
||||||
|
<a v-if="visibility == 'password'" class="btn btn-outline-primary" @click="showInput()">
|
||||||
|
<font-awesome-icon icon="eye" />
|
||||||
|
</a>
|
||||||
|
<a v-if="visibility == 'text'" class="btn btn-outline-primary" @click="hideInput()">
|
||||||
|
<font-awesome-icon icon="eye-slash" />
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
props: {
|
||||||
|
modelValue: {
|
||||||
|
type: String,
|
||||||
|
default: ""
|
||||||
|
},
|
||||||
|
placeholder: {
|
||||||
|
type: String,
|
||||||
|
default: ""
|
||||||
|
},
|
||||||
|
maxlength: {
|
||||||
|
type: Number,
|
||||||
|
default: 255
|
||||||
|
},
|
||||||
|
autocomplete: {
|
||||||
|
type: Boolean,
|
||||||
|
},
|
||||||
|
required: {
|
||||||
|
type: Boolean
|
||||||
|
},
|
||||||
|
readonly: {
|
||||||
|
type: Boolean,
|
||||||
|
default: false,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
visibility: "password",
|
||||||
|
readOnlyValue: false,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
model: {
|
||||||
|
get() {
|
||||||
|
return this.modelValue
|
||||||
|
},
|
||||||
|
set(value) {
|
||||||
|
this.$emit("update:modelValue", value)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
isReadOnly() {
|
||||||
|
// Actually readonly from prop
|
||||||
|
if (this.readonly) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Hack - Disable Chrome save password
|
||||||
|
return this.readOnlyValue;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
created() {
|
||||||
|
// Hack - Disable Chrome save password
|
||||||
|
if (this.autocomplete) {
|
||||||
|
this.readOnlyValue = "readonly";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
showInput() {
|
||||||
|
this.visibility = "text";
|
||||||
|
},
|
||||||
|
hideInput() {
|
||||||
|
this.visibility = "password";
|
||||||
|
},
|
||||||
|
|
||||||
|
// Hack - Disable Chrome save password
|
||||||
|
removeReadOnly() {
|
||||||
|
if (this.autocomplete) {
|
||||||
|
this.readOnlyValue = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
|
@ -40,7 +40,7 @@
|
||||||
<template v-if="notification.type === 'telegram'">
|
<template v-if="notification.type === 'telegram'">
|
||||||
<div class="mb-3">
|
<div class="mb-3">
|
||||||
<label for="telegram-bot-token" class="form-label">Bot Token</label>
|
<label for="telegram-bot-token" class="form-label">Bot Token</label>
|
||||||
<input id="telegram-bot-token" v-model="notification.telegramBotToken" type="text" class="form-control" required>
|
<HiddenInput id="telegram-bot-token" v-model="notification.telegramBotToken" :required="true" :readonly="true"></HiddenInput>
|
||||||
<div class="form-text">
|
<div class="form-text">
|
||||||
You can get a token from <a href="https://t.me/BotFather" target="_blank">https://t.me/BotFather</a>.
|
You can get a token from <a href="https://t.me/BotFather" target="_blank">https://t.me/BotFather</a>.
|
||||||
</div>
|
</div>
|
||||||
|
@ -130,7 +130,7 @@
|
||||||
|
|
||||||
<div class="mb-3">
|
<div class="mb-3">
|
||||||
<label for="password" class="form-label">Password</label>
|
<label for="password" class="form-label">Password</label>
|
||||||
<input id="password" v-model="notification.smtpPassword" type="password" class="form-control" autocomplete="false">
|
<HiddenInput id="password" v-model="notification.smtpPassword" :required="true" autocomplete="false"></HiddenInput>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="mb-3">
|
<div class="mb-3">
|
||||||
|
@ -195,7 +195,7 @@
|
||||||
<template v-if="notification.type === 'gotify'">
|
<template v-if="notification.type === 'gotify'">
|
||||||
<div class="mb-3">
|
<div class="mb-3">
|
||||||
<label for="gotify-application-token" class="form-label">Application Token</label>
|
<label for="gotify-application-token" class="form-label">Application Token</label>
|
||||||
<input id="gotify-application-token" v-model="notification.gotifyapplicationToken" type="text" class="form-control" required>
|
<HiddenInput id="gotify-application-token" v-model="notification.gotifyapplicationToken" :required="true"></HiddenInput>
|
||||||
</div>
|
</div>
|
||||||
<div class="mb-3">
|
<div class="mb-3">
|
||||||
<label for="gotify-server-url" class="form-label">Server URL</label>
|
<label for="gotify-server-url" class="form-label">Server URL</label>
|
||||||
|
@ -306,13 +306,13 @@
|
||||||
<template v-if="notification.type === 'pushy'">
|
<template v-if="notification.type === 'pushy'">
|
||||||
<div class="mb-3">
|
<div class="mb-3">
|
||||||
<label for="pushy-app-token" class="form-label">API_KEY</label>
|
<label for="pushy-app-token" class="form-label">API_KEY</label>
|
||||||
<input id="pushy-app-token" v-model="notification.pushyAPIKey" type="text" class="form-control" required>
|
<HiddenInput id="pushy-app-token" v-model="notification.pushyAPIKey" :required="true"></HiddenInput>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="mb-3">
|
<div class="mb-3">
|
||||||
<label for="pushy-user-key" class="form-label">USER_TOKEN</label>
|
<label for="pushy-user-key" class="form-label">USER_TOKEN</label>
|
||||||
<div class="input-group mb-3">
|
<div class="input-group mb-3">
|
||||||
<input id="pushy-user-key" v-model="notification.pushyToken" type="text" class="form-control" required>
|
<HiddenInput id="pushy-user-key" v-model="notification.pushyToken" :required="true"></HiddenInput>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<p style="margin-top: 8px;">
|
<p style="margin-top: 8px;">
|
||||||
|
@ -323,7 +323,7 @@
|
||||||
<template v-if="notification.type === 'octopush'">
|
<template v-if="notification.type === 'octopush'">
|
||||||
<div class="mb-3">
|
<div class="mb-3">
|
||||||
<label for="octopush-key" class="form-label">API KEY</label>
|
<label for="octopush-key" class="form-label">API KEY</label>
|
||||||
<input id="octopush-key" v-model="notification.octopushAPIKey" type="text" class="form-control" required>
|
<HiddenInput id="octopush-key" v-model="notification.octopushAPIKey" :required="true"></HiddenInput>
|
||||||
<label for="octopush-login" class="form-label">API LOGIN</label>
|
<label for="octopush-login" class="form-label">API LOGIN</label>
|
||||||
<input id="octopush-login" v-model="notification.octopushLogin" type="text" class="form-control" required>
|
<input id="octopush-login" v-model="notification.octopushLogin" type="text" class="form-control" required>
|
||||||
</div>
|
</div>
|
||||||
|
@ -354,9 +354,9 @@
|
||||||
<template v-if="notification.type === 'pushover'">
|
<template v-if="notification.type === 'pushover'">
|
||||||
<div class="mb-3">
|
<div class="mb-3">
|
||||||
<label for="pushover-user" class="form-label">User Key<span style="color: red;"><sup>*</sup></span></label>
|
<label for="pushover-user" class="form-label">User Key<span style="color: red;"><sup>*</sup></span></label>
|
||||||
<input id="pushover-user" v-model="notification.pushoveruserkey" type="text" class="form-control" required>
|
<HiddenInput id="pushover-user" v-model="notification.pushoveruserkey" :required="true"></HiddenInput>
|
||||||
<label for="pushover-app-token" class="form-label">Application Token<span style="color: red;"><sup>*</sup></span></label>
|
<label for="pushover-app-token" class="form-label">Application Token<span style="color: red;"><sup>*</sup></span></label>
|
||||||
<input id="pushover-app-token" v-model="notification.pushoverapptoken" type="text" class="form-control" required>
|
<HiddenInput id="pushover-app-token" v-model="notification.pushoverapptoken" :required="true"></HiddenInput>
|
||||||
<label for="pushover-device" class="form-label">Device</label>
|
<label for="pushover-device" class="form-label">Device</label>
|
||||||
<input id="pushover-device" v-model="notification.pushoverdevice" type="text" class="form-control">
|
<input id="pushover-device" v-model="notification.pushoverdevice" type="text" class="form-control">
|
||||||
<label for="pushover-device" class="form-label">Message Title</label>
|
<label for="pushover-device" class="form-label">Message Title</label>
|
||||||
|
@ -442,7 +442,7 @@
|
||||||
<template v-if="notification.type === 'pushbullet'">
|
<template v-if="notification.type === 'pushbullet'">
|
||||||
<div class="mb-3">
|
<div class="mb-3">
|
||||||
<label for="pushbullet-access-token" class="form-label">Access Token</label>
|
<label for="pushbullet-access-token" class="form-label">Access Token</label>
|
||||||
<input id="pushbullet-access-token" v-model="notification.pushbulletAccessToken" type="text" class="form-control" required>
|
<HiddenInput id="pushbullet-access-token" v-model="notification.pushbulletAccessToken" :required="true"></HiddenInput>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<p style="margin-top: 8px;">
|
<p style="margin-top: 8px;">
|
||||||
|
@ -453,7 +453,7 @@
|
||||||
<template v-if="notification.type === 'line'">
|
<template v-if="notification.type === 'line'">
|
||||||
<div class="mb-3">
|
<div class="mb-3">
|
||||||
<label for="line-channel-access-token" class="form-label">Channel access token</label>
|
<label for="line-channel-access-token" class="form-label">Channel access token</label>
|
||||||
<input id="line-channel-access-token" v-model="notification.lineChannelAccessToken" type="text" class="form-control" required>
|
<HiddenInput id="line-channel-access-token" v-model="notification.lineChannelAccessToken" :required="true"></HiddenInput>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-text">
|
<div class="form-text">
|
||||||
Line Developers Console - <b>Basic Settings</b>
|
Line Developers Console - <b>Basic Settings</b>
|
||||||
|
@ -497,11 +497,13 @@ import { ucfirst } from "../util.ts"
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import { useToast } from "vue-toastification"
|
import { useToast } from "vue-toastification"
|
||||||
import Confirm from "./Confirm.vue";
|
import Confirm from "./Confirm.vue";
|
||||||
|
import HiddenInput from "./HiddenInput.vue";
|
||||||
const toast = useToast()
|
const toast = useToast()
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
components: {
|
components: {
|
||||||
Confirm,
|
Confirm,
|
||||||
|
HiddenInput,
|
||||||
},
|
},
|
||||||
props: {},
|
props: {},
|
||||||
data() {
|
data() {
|
||||||
|
|
|
@ -1,10 +1,10 @@
|
||||||
import { library } from "@fortawesome/fontawesome-svg-core"
|
import { library } from "@fortawesome/fontawesome-svg-core"
|
||||||
import { faCog, faEdit, faPlus, faPause, faPlay, faTachometerAlt, faTrash, faList, faArrowAltCircleUp } from "@fortawesome/free-solid-svg-icons"
|
import { faCog, faEdit, faPlus, faPause, faPlay, faTachometerAlt, faTrash, faList, faArrowAltCircleUp, faEye, faEyeSlash } from "@fortawesome/free-solid-svg-icons"
|
||||||
//import { fa } from '@fortawesome/free-regular-svg-icons'
|
//import { fa } from '@fortawesome/free-regular-svg-icons'
|
||||||
import { FontAwesomeIcon } from "@fortawesome/vue-fontawesome"
|
import { FontAwesomeIcon } from "@fortawesome/vue-fontawesome"
|
||||||
|
|
||||||
// Add Free Font Awesome Icons here
|
// Add Free Font Awesome Icons here
|
||||||
// https://fontawesome.com/v5.15/icons?d=gallery&p=2&s=solid&m=free
|
// https://fontawesome.com/v5.15/icons?d=gallery&p=2&s=solid&m=free
|
||||||
library.add(faCog, faEdit, faPlus, faPause, faPlay, faTachometerAlt, faTrash, faList, faArrowAltCircleUp);
|
library.add(faCog, faEdit, faPlus, faPause, faPlay, faTachometerAlt, faTrash, faList, faArrowAltCircleUp, faEye, faEyeSlash);
|
||||||
|
|
||||||
export { FontAwesomeIcon }
|
export { FontAwesomeIcon }
|
||||||
|
|
|
@ -108,5 +108,12 @@ export default {
|
||||||
"Repeat Password": "Gentag adgangskoden",
|
"Repeat Password": "Gentag adgangskoden",
|
||||||
"Resource Record Type": "Resource Record Type",
|
"Resource Record Type": "Resource Record Type",
|
||||||
respTime: "Resp. Time (ms)",
|
respTime: "Resp. Time (ms)",
|
||||||
notAvailableShort: "N/A"
|
notAvailableShort: "N/A",
|
||||||
|
Create: "Create",
|
||||||
|
clearEventsMsg: "Are you sure want to delete all events for this monitor?",
|
||||||
|
clearHeartbeatsMsg: "Are you sure want to delete all heartbeats for this monitor?",
|
||||||
|
confirmClearStatisticsMsg: "Are you sure want to delete ALL statistics?",
|
||||||
|
"Clear Data": "Clear Data",
|
||||||
|
Events: "Events",
|
||||||
|
Heartbeats: "Heartbeats"
|
||||||
}
|
}
|
||||||
|
|
|
@ -104,6 +104,12 @@ export default {
|
||||||
rrtypeDescription: "Wähle den RR-Typ aus, welchen du überwachen möchtest.",
|
rrtypeDescription: "Wähle den RR-Typ aus, welchen du überwachen möchtest.",
|
||||||
"Last Result": "Letztes Ergebnis",
|
"Last Result": "Letztes Ergebnis",
|
||||||
pauseMonitorMsg: "Bist du sicher das du den Monitor pausieren möchtest?",
|
pauseMonitorMsg: "Bist du sicher das du den Monitor pausieren möchtest?",
|
||||||
|
clearEventsMsg: "Bist du sicher das du alle Ereignisse für diesen Monitor löschen möchtest?",
|
||||||
|
clearHeartbeatsMsg: "Bist du sicher das du alle Statistiken für diesen Monitor löschen möchtest?",
|
||||||
|
"Clear Data": "Lösche Daten",
|
||||||
|
Events: "Ereignisse",
|
||||||
|
Heartbeats: "Statistiken",
|
||||||
|
confirmClearStatisticsMsg: "Bist du sicher das du ALLE Statistiken löschen möchtest?",
|
||||||
"Create your admin account": "Erstelle dein Admin Konto",
|
"Create your admin account": "Erstelle dein Admin Konto",
|
||||||
"Repeat Password": "Wiederhole das Passwort",
|
"Repeat Password": "Wiederhole das Passwort",
|
||||||
"Resource Record Type": "Resource Record Type",
|
"Resource Record Type": "Resource Record Type",
|
||||||
|
@ -111,5 +117,6 @@ export default {
|
||||||
"Export": "Export",
|
"Export": "Export",
|
||||||
"Import": "Import",
|
"Import": "Import",
|
||||||
respTime: "Antw. Zeit (ms)",
|
respTime: "Antw. Zeit (ms)",
|
||||||
notAvailableShort: "N/A"
|
notAvailableShort: "N/A",
|
||||||
|
Create: "Erstellen",
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,6 +16,9 @@ export default {
|
||||||
resoverserverDescription: "Cloudflare is the default server, you can change the resolver server anytime.",
|
resoverserverDescription: "Cloudflare is the default server, you can change the resolver server anytime.",
|
||||||
rrtypeDescription: "Select the RR-Type you want to monitor",
|
rrtypeDescription: "Select the RR-Type you want to monitor",
|
||||||
pauseMonitorMsg: "Are you sure want to pause?",
|
pauseMonitorMsg: "Are you sure want to pause?",
|
||||||
|
clearEventsMsg: "Are you sure want to delete all events for this monitor?",
|
||||||
|
clearHeartbeatsMsg: "Are you sure want to delete all heartbeats for this monitor?",
|
||||||
|
confirmClearStatisticsMsg: "Are you sure want to delete ALL statistics?",
|
||||||
Settings: "Settings",
|
Settings: "Settings",
|
||||||
Dashboard: "Dashboard",
|
Dashboard: "Dashboard",
|
||||||
"New Update": "New Update",
|
"New Update": "New Update",
|
||||||
|
@ -111,5 +114,9 @@ export default {
|
||||||
"Export": "Export",
|
"Export": "Export",
|
||||||
"Import": "Import",
|
"Import": "Import",
|
||||||
respTime: "Resp. Time (ms)",
|
respTime: "Resp. Time (ms)",
|
||||||
notAvailableShort: "N/A"
|
notAvailableShort: "N/A",
|
||||||
|
Create: "Create",
|
||||||
|
"Clear Data": "Clear Data",
|
||||||
|
Events: "Events",
|
||||||
|
Heartbeats: "Heartbeats"
|
||||||
}
|
}
|
||||||
|
|
|
@ -108,5 +108,12 @@ export default {
|
||||||
"Create your admin account": "Crea tu cuenta de administrador",
|
"Create your admin account": "Crea tu cuenta de administrador",
|
||||||
"Repeat Password": "Repetir contraseña",
|
"Repeat Password": "Repetir contraseña",
|
||||||
respTime: "Tiempo de resp. (ms)",
|
respTime: "Tiempo de resp. (ms)",
|
||||||
notAvailableShort: "N/A"
|
notAvailableShort: "N/A",
|
||||||
|
Create: "Create",
|
||||||
|
clearEventsMsg: "Are you sure want to delete all events for this monitor?",
|
||||||
|
clearHeartbeatsMsg: "Are you sure want to delete all heartbeats for this monitor?",
|
||||||
|
confirmClearStatisticsMsg: "Are you sure want to delete ALL statistics?",
|
||||||
|
"Clear Data": "Clear Data",
|
||||||
|
Events: "Events",
|
||||||
|
Heartbeats: "Heartbeats"
|
||||||
}
|
}
|
||||||
|
|
|
@ -66,7 +66,7 @@ export default {
|
||||||
"Theme - Heartbeat Bar": "Voir les services surveillés",
|
"Theme - Heartbeat Bar": "Voir les services surveillés",
|
||||||
Normal: "Général",
|
Normal: "Général",
|
||||||
Bottom: "En dessous",
|
Bottom: "En dessous",
|
||||||
None: "Non",
|
None: "Rien",
|
||||||
Timezone: "Fuseau Horaire",
|
Timezone: "Fuseau Horaire",
|
||||||
"Search Engine Visibility": "Visibilité par les moteurs de recherche",
|
"Search Engine Visibility": "Visibilité par les moteurs de recherche",
|
||||||
"Allow indexing": "Autoriser l'indexation par des moteurs de recherche",
|
"Allow indexing": "Autoriser l'indexation par des moteurs de recherche",
|
||||||
|
@ -108,5 +108,12 @@ export default {
|
||||||
"Create your admin account": "Créez votre compte administrateur",
|
"Create your admin account": "Créez votre compte administrateur",
|
||||||
"Repeat Password": "Répéter le mot de passe",
|
"Repeat Password": "Répéter le mot de passe",
|
||||||
respTime: "Temps de réponse (ms)",
|
respTime: "Temps de réponse (ms)",
|
||||||
notAvailableShort: "N/A"
|
notAvailableShort: "N/A",
|
||||||
|
Create: "Créer",
|
||||||
|
clearEventsMsg: "Are you sure want to delete all events for this monitor?",
|
||||||
|
clearHeartbeatsMsg: "Are you sure want to delete all heartbeats for this monitor?",
|
||||||
|
confirmClearStatisticsMsg: "Are you sure want to delete ALL statistics?",
|
||||||
|
"Clear Data": "Clear Data",
|
||||||
|
Events: "Events",
|
||||||
|
Heartbeats: "Heartbeats"
|
||||||
}
|
}
|
|
@ -108,5 +108,12 @@ export default {
|
||||||
"Create your admin account": "Create your admin account",
|
"Create your admin account": "Create your admin account",
|
||||||
"Repeat Password": "Repeat Password",
|
"Repeat Password": "Repeat Password",
|
||||||
respTime: "Resp. Time (ms)",
|
respTime: "Resp. Time (ms)",
|
||||||
notAvailableShort: "N/A"
|
notAvailableShort: "N/A",
|
||||||
|
Create: "Create",
|
||||||
|
clearEventsMsg: "Are you sure want to delete all events for this monitor?",
|
||||||
|
clearHeartbeatsMsg: "Are you sure want to delete all heartbeats for this monitor?",
|
||||||
|
confirmClearStatisticsMsg: "Are you sure want to delete ALL statistics?",
|
||||||
|
"Clear Data": "Clear Data",
|
||||||
|
Events: "Events",
|
||||||
|
Heartbeats: "Heartbeats"
|
||||||
}
|
}
|
||||||
|
|
|
@ -108,5 +108,12 @@ export default {
|
||||||
"Create your admin account": "관리자 계정 만들기",
|
"Create your admin account": "관리자 계정 만들기",
|
||||||
"Repeat Password": "비밀번호 재입력",
|
"Repeat Password": "비밀번호 재입력",
|
||||||
respTime: "응답 시간 (ms)",
|
respTime: "응답 시간 (ms)",
|
||||||
notAvailableShort: "N/A"
|
notAvailableShort: "N/A",
|
||||||
|
Create: "Create",
|
||||||
|
clearEventsMsg: "Are you sure want to delete all events for this monitor?",
|
||||||
|
clearHeartbeatsMsg: "Are you sure want to delete all heartbeats for this monitor?",
|
||||||
|
confirmClearStatisticsMsg: "Are you sure want to delete ALL statistics?",
|
||||||
|
"Clear Data": "Clear Data",
|
||||||
|
Events: "Events",
|
||||||
|
Heartbeats: "Heartbeats"
|
||||||
}
|
}
|
||||||
|
|
|
@ -108,5 +108,12 @@ export default {
|
||||||
"Create your admin account": "Maak uw beheerdersaccount aan",
|
"Create your admin account": "Maak uw beheerdersaccount aan",
|
||||||
"Repeat Password": "Herhaal wachtwoord",
|
"Repeat Password": "Herhaal wachtwoord",
|
||||||
respTime: "resp. tijd (ms)",
|
respTime: "resp. tijd (ms)",
|
||||||
notAvailableShort: "N.v.t."
|
notAvailableShort: "N.v.t.",
|
||||||
|
Create: "Create",
|
||||||
|
clearEventsMsg: "Are you sure want to delete all events for this monitor?",
|
||||||
|
clearHeartbeatsMsg: "Are you sure want to delete all heartbeats for this monitor?",
|
||||||
|
confirmClearStatisticsMsg: "Are you sure want to delete ALL statistics?",
|
||||||
|
"Clear Data": "Clear Data",
|
||||||
|
Events: "Events",
|
||||||
|
Heartbeats: "Heartbeats"
|
||||||
}
|
}
|
|
@ -0,0 +1,113 @@
|
||||||
|
export default {
|
||||||
|
languageName: "Polski",
|
||||||
|
checkEverySecond: "Sprawdzaj co {0} sekund.",
|
||||||
|
"Avg.": "Średnia ",
|
||||||
|
retriesDescription: "Maksymalna liczba powtórzeń, zanim usługa zostanie oznaczona jako wyłączona i zostanie wysłane powiadomienie",
|
||||||
|
ignoreTLSError: "Ignoruj błąd TLS/SSL dla stron HTTPS",
|
||||||
|
upsideDownModeDescription: "Odwróć status do góry nogami. Jeśli usługa jest osiągalna, to jest oznaczona jako niedostępna.",
|
||||||
|
maxRedirectDescription: "Maksymalna liczba przekierowań do wykonania. Ustaw na 0, aby wyłączyć przekierowania.",
|
||||||
|
acceptedStatusCodesDescription: "Wybierz kody stanu, które są uważane za udaną odpowiedź.",
|
||||||
|
passwordNotMatchMsg: "Powtórzone hasło nie pasuje.",
|
||||||
|
notificationDescription: "Proszę przypisać powiadomienie do monitora(ów), aby zadziałało.",
|
||||||
|
keywordDescription: "Wyszukiwanie słów kluczowych w zwykłym html lub odpowiedzi JSON. Wielkość liter ma znaczenie.",
|
||||||
|
pauseDashboardHome: "Pauza",
|
||||||
|
deleteMonitorMsg: "Czy na pewno chcesz usunąć ten monitor?",
|
||||||
|
deleteNotificationMsg: "Czy na pewno chcesz usunąć to powiadomienie dla wszystkich monitorów?",
|
||||||
|
resoverserverDescription: "Cloudflare jest domyślnym serwerem, możesz zmienić serwer resolver w każdej chwili.",
|
||||||
|
rrtypeDescription: "Wybierz RR-Type który chcesz monitorować",
|
||||||
|
pauseMonitorMsg: "Czy na pewno chcesz wstrzymać?",
|
||||||
|
Settings: "Ustawienia",
|
||||||
|
Dashboard: "Panel",
|
||||||
|
"New Update": "Nowa aktualizacja",
|
||||||
|
Language: "Język",
|
||||||
|
Appearance: "Wygląd",
|
||||||
|
Theme: "Motyw",
|
||||||
|
General: "Ogólne",
|
||||||
|
Version: "Wersja",
|
||||||
|
"Check Update On GitHub": "Sprawdź aktualizację na GitHub.",
|
||||||
|
List: "Lista",
|
||||||
|
Add: "Dodaj",
|
||||||
|
"Add New Monitor": "Dodaj nowy monitor",
|
||||||
|
"Quick Stats": "Szybkie statystyki",
|
||||||
|
Up: "Online",
|
||||||
|
Down: "Offline",
|
||||||
|
Pending: "Oczekujący",
|
||||||
|
Unknown: "Nieznane",
|
||||||
|
Pause: "Pauza",
|
||||||
|
Name: "Nazwa",
|
||||||
|
Status: "Status",
|
||||||
|
DateTime: "Data i godzina",
|
||||||
|
Message: "Wiadomość",
|
||||||
|
"No important events": "Brak ważnych wydarzeń",
|
||||||
|
Resume: "Wznów",
|
||||||
|
Edit: "Edytuj",
|
||||||
|
Delete: "Usuń",
|
||||||
|
Current: "aktualny",
|
||||||
|
Uptime: "Czas pracy",
|
||||||
|
"Cert Exp.": "Wygaśnięcie certyfikatu",
|
||||||
|
days: "dni",
|
||||||
|
day: "dzień",
|
||||||
|
"-day": " dni",
|
||||||
|
hour: "godzina",
|
||||||
|
"-hour": " godziny",
|
||||||
|
Response: "Odpowiedź",
|
||||||
|
Ping: "Ping",
|
||||||
|
"Monitor Type": "Typ monitora",
|
||||||
|
Keyword: "Słowo kluczowe",
|
||||||
|
"Friendly Name": "Przyjazna nazwa",
|
||||||
|
URL: "URL",
|
||||||
|
Hostname: "Nazwa hosta",
|
||||||
|
Port: "Port",
|
||||||
|
"Heartbeat Interval": "Interwał bicia serca",
|
||||||
|
Retries: "Prób",
|
||||||
|
Advanced: "Zaawansowane",
|
||||||
|
"Upside Down Mode": "Tryb do góry nogami",
|
||||||
|
"Max. Redirects": "Maks. przekierowania",
|
||||||
|
"Accepted Status Codes": "Akceptowane kody statusu",
|
||||||
|
Save: "Zapisz",
|
||||||
|
Notifications: "Powiadomienia",
|
||||||
|
"Not available, please setup.": "Niedostępne, proszę skonfigurować.",
|
||||||
|
"Setup Notification": "Konfiguracja powiadomień",
|
||||||
|
Light: "Jasny",
|
||||||
|
Dark: "Ciemny",
|
||||||
|
Auto: "Automatyczny",
|
||||||
|
"Theme - Heartbeat Bar": "Motyw - pasek bicia serca",
|
||||||
|
Normal: "Normalne",
|
||||||
|
Bottom: "Na dole",
|
||||||
|
None: "Brak",
|
||||||
|
Timezone: "Strefa czasowa",
|
||||||
|
"Search Engine Visibility": "Widoczność w wyszukiwarce",
|
||||||
|
"Allow indexing": "Pozwól na indeksowanie",
|
||||||
|
"Discourage search engines from indexing site": "Zniechęcaj wyszukiwarki do indeksowania strony",
|
||||||
|
"Change Password": "Zmień hasło",
|
||||||
|
"Current Password": "Aktualne hasło",
|
||||||
|
"New Password": "Nowe hasło",
|
||||||
|
"Repeat New Password": "Powtórz nowe hasło",
|
||||||
|
"Update Password": "Zaktualizuj hasło",
|
||||||
|
"Disable Auth": "Wyłącz autoryzację",
|
||||||
|
"Enable Auth": "Włącz autoryzację ",
|
||||||
|
Logout: "Wyloguj się",
|
||||||
|
Leave: "Zostaw",
|
||||||
|
"I understand, please disable": "Rozumiem, proszę wyłączyć",
|
||||||
|
Confirm: "Potwierdź",
|
||||||
|
Yes: "Tak",
|
||||||
|
No: "Nie",
|
||||||
|
Username: "Nazwa użytkownika",
|
||||||
|
Password: "Hasło",
|
||||||
|
"Remember me": "Zapamiętaj mnie",
|
||||||
|
Login: "Zaloguj się",
|
||||||
|
"No Monitors, please": "Brak monitorów, proszę",
|
||||||
|
"add one": "dodaj jeden",
|
||||||
|
"Notification Type": "Typ powiadomienia",
|
||||||
|
Email: "Email",
|
||||||
|
Test: "Test",
|
||||||
|
"Certificate Info": "Informacje o certyfikacie",
|
||||||
|
"Resolver Server": "Server resolver",
|
||||||
|
"Resource Record Type": "Typ rekordu zasobów",
|
||||||
|
"Last Result": "Ostatni wynik",
|
||||||
|
"Create your admin account": "Utwórz swoje konto administratora",
|
||||||
|
"Repeat Password": "Powtórz hasło",
|
||||||
|
respTime: "Czas odp. (ms)",
|
||||||
|
notAvailableShort: "N/A",
|
||||||
|
Create: "Stwórz"
|
||||||
|
}
|
|
@ -108,5 +108,12 @@ export default {
|
||||||
"Create your admin account": "Создайте аккаунт администратора",
|
"Create your admin account": "Создайте аккаунт администратора",
|
||||||
"Repeat Password": "Повторите пароль",
|
"Repeat Password": "Повторите пароль",
|
||||||
respTime: "Resp. Time (ms)",
|
respTime: "Resp. Time (ms)",
|
||||||
notAvailableShort: "N/A"
|
notAvailableShort: "N/A",
|
||||||
|
Create: "Create",
|
||||||
|
clearEventsMsg: "Are you sure want to delete all events for this monitor?",
|
||||||
|
clearHeartbeatsMsg: "Are you sure want to delete all heartbeats for this monitor?",
|
||||||
|
confirmClearStatisticsMsg: "Are you sure want to delete ALL statistics?",
|
||||||
|
"Clear Data": "Clear Data",
|
||||||
|
Events: "Events",
|
||||||
|
Heartbeats: "Heartbeats"
|
||||||
}
|
}
|
||||||
|
|
|
@ -108,5 +108,12 @@ export default {
|
||||||
"Create your admin account": "Naprivi administratorski nalog",
|
"Create your admin account": "Naprivi administratorski nalog",
|
||||||
"Repeat Password": "Ponovite lozinku",
|
"Repeat Password": "Ponovite lozinku",
|
||||||
respTime: "Vreme odg. (ms)",
|
respTime: "Vreme odg. (ms)",
|
||||||
notAvailableShort: "N/A"
|
notAvailableShort: "N/A",
|
||||||
|
Create: "Create",
|
||||||
|
clearEventsMsg: "Are you sure want to delete all events for this monitor?",
|
||||||
|
clearHeartbeatsMsg: "Are you sure want to delete all heartbeats for this monitor?",
|
||||||
|
confirmClearStatisticsMsg: "Are you sure want to delete ALL statistics?",
|
||||||
|
"Clear Data": "Clear Data",
|
||||||
|
Events: "Events",
|
||||||
|
Heartbeats: "Heartbeats"
|
||||||
}
|
}
|
||||||
|
|
|
@ -108,5 +108,12 @@ export default {
|
||||||
"Create your admin account": "Наприви администраторски налог",
|
"Create your admin account": "Наприви администраторски налог",
|
||||||
"Repeat Password": "Поновите лозинку",
|
"Repeat Password": "Поновите лозинку",
|
||||||
respTime: "Време одг. (мс)",
|
respTime: "Време одг. (мс)",
|
||||||
notAvailableShort: "N/A"
|
notAvailableShort: "N/A",
|
||||||
|
Create: "Create",
|
||||||
|
clearEventsMsg: "Are you sure want to delete all events for this monitor?",
|
||||||
|
clearHeartbeatsMsg: "Are you sure want to delete all heartbeats for this monitor?",
|
||||||
|
confirmClearStatisticsMsg: "Are you sure want to delete ALL statistics?",
|
||||||
|
"Clear Data": "Clear Data",
|
||||||
|
Events: "Events",
|
||||||
|
Heartbeats: "Heartbeats"
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,14 +1,14 @@
|
||||||
export default {
|
export default {
|
||||||
languageName: "Swedish",
|
languageName: "Svenska",
|
||||||
checkEverySecond: "Uppdatera var {0} sekund.",
|
checkEverySecond: "Uppdatera var {0} sekund.",
|
||||||
"Avg.": "Genomsnitt ",
|
"Avg.": "Genomsnittligt ",
|
||||||
retriesDescription: "Max antal försök innan tjänsten markeras som nere och en notis skickas",
|
retriesDescription: "Max antal försök innan tjänsten markeras som nere och en notis skickas",
|
||||||
ignoreTLSError: "Ignorera TLS/SSL-fel för webbsidor med HTTPS",
|
ignoreTLSError: "Ignorera TLS/SSL-fel för webbsidor med HTTPS",
|
||||||
upsideDownModeDescription: "Vänd upp och ner på statusen. Om tjänsten är nåbar visas den som NERE.",
|
upsideDownModeDescription: "Vänd upp och ner på statusen. Om tjänsten är nåbar visas den som NERE.",
|
||||||
maxRedirectDescription: "Max antal omdirigeringar att följa. Välj 0 för att avaktivera omdirigeringar.",
|
maxRedirectDescription: "Max antal omdirigeringar att följa. Välj 0 för att avaktivera omdirigeringar.",
|
||||||
acceptedStatusCodesDescription: "Välj statuskoder som räknas som lyckade.",
|
acceptedStatusCodesDescription: "Välj statuskoder som räknas som lyckade.",
|
||||||
passwordNotMatchMsg: "Det bekräftade lösenordet stämmer ej överens.",
|
passwordNotMatchMsg: "Det bekräftade lösenordet stämmer ej överens.",
|
||||||
notificationDescription: "Vänligen lägg till en notistjänst till övervakaren.",
|
notificationDescription: "Vänligen lägg till en notistjänst till dina övervakare.",
|
||||||
keywordDescription: "Sök efter nyckelord i ren HTML eller JSON-svar. Sökningen är skiftkänslig.",
|
keywordDescription: "Sök efter nyckelord i ren HTML eller JSON-svar. Sökningen är skiftkänslig.",
|
||||||
pauseDashboardHome: "Pausa",
|
pauseDashboardHome: "Pausa",
|
||||||
deleteMonitorMsg: "Är du säker på att du vill ta bort den här övervakningen?",
|
deleteMonitorMsg: "Är du säker på att du vill ta bort den här övervakningen?",
|
||||||
|
@ -33,10 +33,10 @@ export default {
|
||||||
Down: "Nere",
|
Down: "Nere",
|
||||||
Pending: "Pågående",
|
Pending: "Pågående",
|
||||||
Unknown: "Okänt",
|
Unknown: "Okänt",
|
||||||
Pause: "Paus",
|
Pause: "Pausa",
|
||||||
Name: "Namn",
|
Name: "Namn",
|
||||||
Status: "Status",
|
Status: "Status",
|
||||||
DateTime: "DatumTid",
|
DateTime: "Datum & Tid",
|
||||||
Message: "Meddelande",
|
Message: "Meddelande",
|
||||||
"No important events": "Inga viktiga händelser",
|
"No important events": "Inga viktiga händelser",
|
||||||
Resume: "Återuppta",
|
Resume: "Återuppta",
|
||||||
|
@ -44,17 +44,17 @@ export default {
|
||||||
Delete: "Ta bort",
|
Delete: "Ta bort",
|
||||||
Current: "Nuvarande",
|
Current: "Nuvarande",
|
||||||
Uptime: "Drifttid",
|
Uptime: "Drifttid",
|
||||||
"Cert Exp.": "Certifikatsutgång",
|
"Cert Exp.": "Certifikat utgår",
|
||||||
days: "dagar",
|
days: "dagar",
|
||||||
day: "dag",
|
day: "dag",
|
||||||
"-day": "-dag",
|
"-day": " dagar",
|
||||||
hour: "timme",
|
hour: "timme",
|
||||||
"-hour": "-timme",
|
"-hour": " timmar",
|
||||||
Response: "Svar",
|
Response: "Svar",
|
||||||
Ping: "Ping",
|
Ping: "Ping",
|
||||||
"Monitor Type": "Övervakningstyp",
|
"Monitor Type": "Övervakningstyp",
|
||||||
Keyword: "Nyckelord",
|
Keyword: "Nyckelord",
|
||||||
"Friendly Name": "Vänligt Namn",
|
"Friendly Name": "Namn",
|
||||||
URL: "URL",
|
URL: "URL",
|
||||||
Hostname: "Värdnamn",
|
Hostname: "Värdnamn",
|
||||||
Port: "Port",
|
Port: "Port",
|
||||||
|
@ -67,14 +67,14 @@ export default {
|
||||||
Save: "Spara",
|
Save: "Spara",
|
||||||
Notifications: "Notiser",
|
Notifications: "Notiser",
|
||||||
"Not available, please setup.": "Ej tillgänglig, vänligen konfigurera.",
|
"Not available, please setup.": "Ej tillgänglig, vänligen konfigurera.",
|
||||||
"Setup Notification": "Konfigurera Notis",
|
"Setup Notification": "Ny Notistjänst",
|
||||||
Light: "Ljust",
|
Light: "Ljust",
|
||||||
Dark: "Mörkt",
|
Dark: "Mörkt",
|
||||||
Auto: "Automatisk",
|
Auto: "Automatiskt",
|
||||||
"Theme - Heartbeat Bar": "Tema - Heartbeat Bar",
|
"Theme - Heartbeat Bar": "Tema - Heartbeat Bar",
|
||||||
Normal: "Normal",
|
Normal: "Normal",
|
||||||
Bottom: "Botten",
|
Bottom: "Botten",
|
||||||
None: "Ingen",
|
None: "Tomt",
|
||||||
Timezone: "Tidszon",
|
Timezone: "Tidszon",
|
||||||
"Search Engine Visibility": "Synlighet på Sökmotorer",
|
"Search Engine Visibility": "Synlighet på Sökmotorer",
|
||||||
"Allow indexing": "Tillåt indexering",
|
"Allow indexing": "Tillåt indexering",
|
||||||
|
@ -107,6 +107,13 @@ export default {
|
||||||
"Last Result": "Senaste resultat",
|
"Last Result": "Senaste resultat",
|
||||||
"Create your admin account": "Skapa ditt administratörskonto",
|
"Create your admin account": "Skapa ditt administratörskonto",
|
||||||
"Repeat Password": "Upprepa Lösenord",
|
"Repeat Password": "Upprepa Lösenord",
|
||||||
respTime: "Resp. Time (ms)",
|
respTime: "Svarstid (ms)",
|
||||||
notAvailableShort: "N/A"
|
notAvailableShort: "Ej Tillg.",
|
||||||
|
Create: "Create",
|
||||||
|
clearEventsMsg: "Are you sure want to delete all events for this monitor?",
|
||||||
|
clearHeartbeatsMsg: "Are you sure want to delete all heartbeats for this monitor?",
|
||||||
|
confirmClearStatisticsMsg: "Are you sure want to delete ALL statistics?",
|
||||||
|
"Clear Data": "Clear Data",
|
||||||
|
Events: "Events",
|
||||||
|
Heartbeats: "Heartbeats"
|
||||||
}
|
}
|
||||||
|
|
|
@ -108,5 +108,12 @@ export default {
|
||||||
"Create your admin account": "创建管理员账号",
|
"Create your admin account": "创建管理员账号",
|
||||||
"Repeat Password": "重复密码",
|
"Repeat Password": "重复密码",
|
||||||
respTime: "Resp. Time (ms)",
|
respTime: "Resp. Time (ms)",
|
||||||
notAvailableShort: "N/A"
|
notAvailableShort: "N/A",
|
||||||
|
Create: "Create",
|
||||||
|
clearEventsMsg: "Are you sure want to delete all events for this monitor?",
|
||||||
|
clearHeartbeatsMsg: "Are you sure want to delete all heartbeats for this monitor?",
|
||||||
|
confirmClearStatisticsMsg: "Are you sure want to delete ALL statistics?",
|
||||||
|
"Clear Data": "Clear Data",
|
||||||
|
Events: "Events",
|
||||||
|
Heartbeats: "Heartbeats"
|
||||||
}
|
}
|
||||||
|
|
|
@ -108,5 +108,12 @@ export default {
|
||||||
"Create your admin account": "製作你的管理員帳號",
|
"Create your admin account": "製作你的管理員帳號",
|
||||||
"Repeat Password": "重複密碼",
|
"Repeat Password": "重複密碼",
|
||||||
respTime: "反應時間 (ms)",
|
respTime: "反應時間 (ms)",
|
||||||
notAvailableShort: "N/A"
|
notAvailableShort: "N/A",
|
||||||
|
Create: "建立",
|
||||||
|
clearEventsMsg: "是否確定刪除這個監測器的所有事件?",
|
||||||
|
clearHeartbeatsMsg: "是否確定刪除這個監測器的所有脈搏資料?",
|
||||||
|
confirmClearStatisticsMsg: "是否確定刪除所有監測器的脈搏資料?(您的監測器會繼續正常運作)",
|
||||||
|
"Clear Data": "清除資料",
|
||||||
|
Events: "事件",
|
||||||
|
Heartbeats: "脈搏"
|
||||||
}
|
}
|
||||||
|
|
|
@ -28,7 +28,7 @@ import zhHK from "./languages/zh-HK";
|
||||||
import deDE from "./languages/de-DE";
|
import deDE from "./languages/de-DE";
|
||||||
import nlNL from "./languages/nl-NL";
|
import nlNL from "./languages/nl-NL";
|
||||||
import esEs from "./languages/es-ES";
|
import esEs from "./languages/es-ES";
|
||||||
import fr from "./languages/fr";
|
import frFR from "./languages/fr-FR";
|
||||||
import ja from "./languages/ja";
|
import ja from "./languages/ja";
|
||||||
import daDK from "./languages/da-DK";
|
import daDK from "./languages/da-DK";
|
||||||
import sr from "./languages/sr";
|
import sr from "./languages/sr";
|
||||||
|
@ -37,6 +37,7 @@ import svSE from "./languages/sv-SE";
|
||||||
import koKR from "./languages/ko-KR";
|
import koKR from "./languages/ko-KR";
|
||||||
import ruRU from "./languages/ru-RU";
|
import ruRU from "./languages/ru-RU";
|
||||||
import zhCN from "./languages/zh-CN";
|
import zhCN from "./languages/zh-CN";
|
||||||
|
import pl from "./languages/pl"
|
||||||
|
|
||||||
const routes = [
|
const routes = [
|
||||||
{
|
{
|
||||||
|
@ -105,7 +106,7 @@ const languageList = {
|
||||||
"de-DE": deDE,
|
"de-DE": deDE,
|
||||||
"nl-NL": nlNL,
|
"nl-NL": nlNL,
|
||||||
"es-ES": esEs,
|
"es-ES": esEs,
|
||||||
"fr": fr,
|
"fr-FR": frFR,
|
||||||
"ja": ja,
|
"ja": ja,
|
||||||
"da-DK": daDK,
|
"da-DK": daDK,
|
||||||
"sr": sr,
|
"sr": sr,
|
||||||
|
@ -114,6 +115,7 @@ const languageList = {
|
||||||
"ko-KR": koKR,
|
"ko-KR": koKR,
|
||||||
"ru-RU": ruRU,
|
"ru-RU": ruRU,
|
||||||
"zh-CN": zhCN,
|
"zh-CN": zhCN,
|
||||||
|
"pl": pl,
|
||||||
};
|
};
|
||||||
|
|
||||||
const i18n = createI18n({
|
const i18n = createI18n({
|
||||||
|
|
|
@ -107,8 +107,8 @@ export default {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
socket.on("heartbeatList", (monitorID, data) => {
|
socket.on("heartbeatList", (monitorID, data, overwrite = false) => {
|
||||||
if (! (monitorID in this.heartbeatList)) {
|
if (! (monitorID in this.heartbeatList) || overwrite) {
|
||||||
this.heartbeatList[monitorID] = data;
|
this.heartbeatList[monitorID] = data;
|
||||||
} else {
|
} else {
|
||||||
this.heartbeatList[monitorID] = data.concat(this.heartbeatList[monitorID])
|
this.heartbeatList[monitorID] = data.concat(this.heartbeatList[monitorID])
|
||||||
|
@ -127,8 +127,8 @@ export default {
|
||||||
this.certInfoList[monitorID] = JSON.parse(data)
|
this.certInfoList[monitorID] = JSON.parse(data)
|
||||||
});
|
});
|
||||||
|
|
||||||
socket.on("importantHeartbeatList", (monitorID, data) => {
|
socket.on("importantHeartbeatList", (monitorID, data, overwrite) => {
|
||||||
if (! (monitorID in this.importantHeartbeatList)) {
|
if (! (monitorID in this.importantHeartbeatList) || overwrite) {
|
||||||
this.importantHeartbeatList[monitorID] = data;
|
this.importantHeartbeatList[monitorID] = data;
|
||||||
} else {
|
} else {
|
||||||
this.importantHeartbeatList[monitorID] = data.concat(this.importantHeartbeatList[monitorID])
|
this.importantHeartbeatList[monitorID] = data.concat(this.importantHeartbeatList[monitorID])
|
||||||
|
@ -257,6 +257,18 @@ export default {
|
||||||
uploadBackup(uploadedJSON, callback) {
|
uploadBackup(uploadedJSON, callback) {
|
||||||
socket.emit("uploadBackup", uploadedJSON, callback)
|
socket.emit("uploadBackup", uploadedJSON, callback)
|
||||||
},
|
},
|
||||||
|
|
||||||
|
clearEvents(monitorID, callback) {
|
||||||
|
socket.emit("clearEvents", monitorID, callback)
|
||||||
|
},
|
||||||
|
|
||||||
|
clearHeartbeats(monitorID, callback) {
|
||||||
|
socket.emit("clearHeartbeats", monitorID, callback)
|
||||||
|
},
|
||||||
|
|
||||||
|
clearStatistics(callback) {
|
||||||
|
socket.emit("clearStatistics", callback)
|
||||||
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
computed: {
|
computed: {
|
||||||
|
|
|
@ -11,7 +11,7 @@ export default {
|
||||||
mounted() {
|
mounted() {
|
||||||
// Default Light
|
// Default Light
|
||||||
if (! this.userTheme) {
|
if (! this.userTheme) {
|
||||||
this.userTheme = "light";
|
this.userTheme = "auto";
|
||||||
}
|
}
|
||||||
|
|
||||||
// Default Heartbeat Bar
|
// Default Heartbeat Bar
|
||||||
|
|
|
@ -133,6 +133,23 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="shadow-box table-shadow-box">
|
<div class="shadow-box table-shadow-box">
|
||||||
|
<div class="dropdown dropdown-clear-data">
|
||||||
|
<button class="btn btn-sm btn-outline-danger dropdown-toggle" type="button" data-bs-toggle="dropdown">
|
||||||
|
<font-awesome-icon icon="trash" /> {{ $t("Clear Data") }}
|
||||||
|
</button>
|
||||||
|
<ul class="dropdown-menu dropdown-menu-end">
|
||||||
|
<li>
|
||||||
|
<button type="button" class="dropdown-item" @click="clearEventsDialog">
|
||||||
|
{{ $t("Events") }}
|
||||||
|
</button>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<button type="button" class="dropdown-item" @click="clearHeartbeatsDialog">
|
||||||
|
{{ $t("Heartbeats") }}
|
||||||
|
</button>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
<table class="table table-borderless table-hover">
|
<table class="table table-borderless table-hover">
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
|
@ -172,6 +189,14 @@
|
||||||
<Confirm ref="confirmDelete" btn-style="btn-danger" :yes-text="$t('Yes')" :no-text="$t('No')" @yes="deleteMonitor">
|
<Confirm ref="confirmDelete" btn-style="btn-danger" :yes-text="$t('Yes')" :no-text="$t('No')" @yes="deleteMonitor">
|
||||||
{{ $t("deleteMonitorMsg") }}
|
{{ $t("deleteMonitorMsg") }}
|
||||||
</Confirm>
|
</Confirm>
|
||||||
|
|
||||||
|
<Confirm ref="confirmClearEvents" btn-style="btn-danger" :yes-text="$t('Yes')" :no-text="$t('No')" @yes="clearEvents">
|
||||||
|
{{ $t("clearEventsMsg") }}
|
||||||
|
</Confirm>
|
||||||
|
|
||||||
|
<Confirm ref="confirmClearHeartbeats" btn-style="btn-danger" :yes-text="$t('Yes')" :no-text="$t('No')" @yes="clearHeartbeats">
|
||||||
|
{{ $t("clearHeartbeatsMsg") }}
|
||||||
|
</Confirm>
|
||||||
</div>
|
</div>
|
||||||
</transition>
|
</transition>
|
||||||
</template>
|
</template>
|
||||||
|
@ -251,6 +276,7 @@ export default {
|
||||||
|
|
||||||
importantHeartBeatList() {
|
importantHeartBeatList() {
|
||||||
if (this.$root.importantHeartbeatList[this.monitor.id]) {
|
if (this.$root.importantHeartbeatList[this.monitor.id]) {
|
||||||
|
// eslint-disable-next-line vue/no-side-effects-in-computed-properties
|
||||||
this.heartBeatList = this.$root.importantHeartbeatList[this.monitor.id];
|
this.heartBeatList = this.$root.importantHeartbeatList[this.monitor.id];
|
||||||
return this.$root.importantHeartbeatList[this.monitor.id]
|
return this.$root.importantHeartbeatList[this.monitor.id]
|
||||||
}
|
}
|
||||||
|
@ -313,6 +339,14 @@ export default {
|
||||||
this.$refs.confirmDelete.show();
|
this.$refs.confirmDelete.show();
|
||||||
},
|
},
|
||||||
|
|
||||||
|
clearEventsDialog() {
|
||||||
|
this.$refs.confirmClearEvents.show();
|
||||||
|
},
|
||||||
|
|
||||||
|
clearHeartbeatsDialog() {
|
||||||
|
this.$refs.confirmClearHeartbeats.show();
|
||||||
|
},
|
||||||
|
|
||||||
deleteMonitor() {
|
deleteMonitor() {
|
||||||
this.$root.deleteMonitor(this.monitor.id, (res) => {
|
this.$root.deleteMonitor(this.monitor.id, (res) => {
|
||||||
if (res.ok) {
|
if (res.ok) {
|
||||||
|
@ -324,6 +358,21 @@ export default {
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
|
||||||
|
clearEvents() {
|
||||||
|
this.$root.clearEvents(this.monitor.id, (res) => {
|
||||||
|
if (! res.ok) {
|
||||||
|
toast.error(res.msg);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
},
|
||||||
|
|
||||||
|
clearHeartbeats() {
|
||||||
|
this.$root.clearHeartbeats(this.monitor.id, (res) => {
|
||||||
|
if (! res.ok) {
|
||||||
|
toast.error(res.msg);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
@ -340,16 +389,20 @@ export default {
|
||||||
@media (max-width: 550px) {
|
@media (max-width: 550px) {
|
||||||
.functions {
|
.functions {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
|
||||||
|
|
||||||
button, a {
|
button, a {
|
||||||
margin-left: 10px !important;
|
margin-left: 10px !important;
|
||||||
margin-right: 10px !important;
|
margin-right: 10px !important;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.ping-chart-wrapper {
|
.ping-chart-wrapper {
|
||||||
padding: 10px !important;
|
padding: 10px !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.dropdown-clear-data {
|
||||||
|
margin-bottom: 10px;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@media (max-width: 400px) {
|
@media (max-width: 400px) {
|
||||||
|
@ -364,6 +417,13 @@ export default {
|
||||||
padding-left: 25px;
|
padding-left: 25px;
|
||||||
padding-right: 25px;
|
padding-right: 25px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.dropdown-clear-data {
|
||||||
|
button {
|
||||||
|
display: block;
|
||||||
|
padding-top: 4px;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.url {
|
.url {
|
||||||
|
@ -417,10 +477,30 @@ table {
|
||||||
color: black;
|
color: black;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.dropdown-clear-data {
|
||||||
|
float: right;
|
||||||
|
}
|
||||||
|
|
||||||
.dark {
|
.dark {
|
||||||
.keyword {
|
.keyword {
|
||||||
color: $dark-font-color;
|
color: $dark-font-color;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.dropdown-clear-data {
|
||||||
|
ul {
|
||||||
|
background-color: $dark-bg;
|
||||||
|
border-color: $dark-bg2;
|
||||||
|
border-width: 2px;
|
||||||
|
|
||||||
|
li button {
|
||||||
|
color: $dark-font-color;
|
||||||
|
}
|
||||||
|
|
||||||
|
li button:hover {
|
||||||
|
background-color: $dark-bg2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -134,6 +134,7 @@
|
||||||
<button v-if="settings.disableAuth" class="btn btn-outline-primary me-1" @click="enableAuth">{{ $t("Enable Auth") }}</button>
|
<button v-if="settings.disableAuth" class="btn btn-outline-primary me-1" @click="enableAuth">{{ $t("Enable Auth") }}</button>
|
||||||
<button v-if="! settings.disableAuth" class="btn btn-primary me-1" @click="confirmDisableAuth">{{ $t("Disable Auth") }}</button>
|
<button v-if="! settings.disableAuth" class="btn btn-primary me-1" @click="confirmDisableAuth">{{ $t("Disable Auth") }}</button>
|
||||||
<button v-if="! settings.disableAuth" class="btn btn-danger me-1" @click="$root.logout">{{ $t("Logout") }}</button>
|
<button v-if="! settings.disableAuth" class="btn btn-danger me-1" @click="$root.logout">{{ $t("Logout") }}</button>
|
||||||
|
<button class="btn btn-outline-danger me-1" @click="confirmClearStatistics">{{ $t("Clear all Statistics") }}</button>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
</div>
|
</div>
|
||||||
|
@ -221,6 +222,15 @@
|
||||||
<p>이 기능은 <strong>Cloudflare Access와 같은 서드파티 인증</strong>을 Uptime Kuma 앞에 둔 사용자를 위한 기능이에요.</p>
|
<p>이 기능은 <strong>Cloudflare Access와 같은 서드파티 인증</strong>을 Uptime Kuma 앞에 둔 사용자를 위한 기능이에요.</p>
|
||||||
<p>신중하게 사용하세요.</p>
|
<p>신중하게 사용하세요.</p>
|
||||||
</template>
|
</template>
|
||||||
|
<template v-if="$i18n.locale === 'pl' ">
|
||||||
|
<p>Czy na pewno chcesz <strong>wyłączyć autoryzację</strong>?</p>
|
||||||
|
<p>Jest przeznaczony dla <strong>kogoś, kto ma autoryzację zewnętrzną</strong> przed Uptime Kuma, taką jak Cloudflare Access.</p>
|
||||||
|
<p>Proszę używać ostrożnie.</p>
|
||||||
|
</template>
|
||||||
|
</Confirm>
|
||||||
|
|
||||||
|
<Confirm ref="confirmClearStatistics" btn-style="btn-danger" :yes-text="$t('Yes')" :no-text="$t('No')" @yes="clearStatistics">
|
||||||
|
{{ $t("confirmClearStatisticsMsg") }}
|
||||||
</Confirm>
|
</Confirm>
|
||||||
</div>
|
</div>
|
||||||
</transition>
|
</transition>
|
||||||
|
@ -320,6 +330,10 @@ export default {
|
||||||
this.$refs.confirmDisableAuth.show();
|
this.$refs.confirmDisableAuth.show();
|
||||||
},
|
},
|
||||||
|
|
||||||
|
confirmClearStatistics() {
|
||||||
|
this.$refs.confirmClearStatistics.show();
|
||||||
|
},
|
||||||
|
|
||||||
disableAuth() {
|
disableAuth() {
|
||||||
this.settings.disableAuth = true;
|
this.settings.disableAuth = true;
|
||||||
this.saveSettings();
|
this.saveSettings();
|
||||||
|
@ -366,6 +380,16 @@ export default {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
clearStatistics() {
|
||||||
|
this.$root.clearStatistics((res) => {
|
||||||
|
if (res.ok) {
|
||||||
|
this.$router.go();
|
||||||
|
} else {
|
||||||
|
toast.error(res.msg);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
|
@ -14,6 +14,15 @@
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<div class="form-floating">
|
<div class="form-floating">
|
||||||
|
<select id="language" v-model="$i18n.locale" class="form-select">
|
||||||
|
<option v-for="(lang, i) in $i18n.availableLocales" :key="`Lang${i}`" :value="lang">
|
||||||
|
{{ $i18n.messages[lang].languageName }}
|
||||||
|
</option>
|
||||||
|
</select>
|
||||||
|
<label for="language" class="form-label">{{ $t("Language") }}</label>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-floating mt-3">
|
||||||
<input id="floatingInput" v-model="username" type="text" class="form-control" placeholder="Username" required>
|
<input id="floatingInput" v-model="username" type="text" class="form-control" placeholder="Username" required>
|
||||||
<label for="floatingInput">{{ $t("Username") }}</label>
|
<label for="floatingInput">{{ $t("Username") }}</label>
|
||||||
</div>
|
</div>
|
||||||
|
@ -29,7 +38,7 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<button class="w-100 btn btn-primary mt-3" type="submit" :disabled="processing">
|
<button class="w-100 btn btn-primary mt-3" type="submit" :disabled="processing">
|
||||||
Create
|
{{ $t("Create") }}
|
||||||
</button>
|
</button>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
|
@ -49,6 +58,11 @@ export default {
|
||||||
repeatPassword: "",
|
repeatPassword: "",
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
watch: {
|
||||||
|
"$i18n.locale"() {
|
||||||
|
localStorage.locale = this.$i18n.locale;
|
||||||
|
},
|
||||||
|
},
|
||||||
mounted() {
|
mounted() {
|
||||||
this.$root.getSocket().emit("needSetup", (needSetup) => {
|
this.$root.getSocket().emit("needSetup", (needSetup) => {
|
||||||
if (! needSetup) {
|
if (! needSetup) {
|
||||||
|
|
Loading…
Reference in New Issue