Compare commits

...

17 Commits

Author SHA1 Message Date
9bde02faf6 Fix envsubst issue 2023-12-15 17:43:18 +08:00
0f52bb78b8 Update Readme Badge 2023-12-13 21:08:01 +08:00
03c7815b58 Update to 1.3.3 2023-12-10 21:21:14 +08:00
80d5c685e5 Update reformat-changelog.ts 2023-12-10 21:20:08 +08:00
a8482ec8ac Merge pull request #257 from UptimeKumaBot/weblate-dockge-dockge
Translations update from Kuma Weblate
2023-12-10 21:17:22 +08:00
07c52ccebb Update from upstream (#271)
* Fix

* Fix

* Fix logout

* Fix logout

* Fix

* WIP
2023-12-10 21:16:23 +08:00
587d2dcaca Fix URLs with env 2023-12-10 13:17:04 +08:00
5f1f3593fd Add pnpm run dev 2023-12-10 13:16:40 +08:00
316c566c76 Update dependencies 2023-12-10 02:39:39 +08:00
d32bd3937f Fix: .env code change is not reactive (#270) 2023-12-10 02:29:05 +08:00
007eac7b58 Envsubst YAML in order to display the correct values in the UI instead of tags (#268)
* WIP

* Add envsubst

* WIP
2023-12-10 00:59:28 +08:00
b945ddea55 Fix check update (#269)
* Fix check-version.ts
2023-12-10 00:59:05 +08:00
9b6b49947c Translated using Weblate (Portuguese (Brazil))
Currently translated at 100.0% (100 of 100 strings)

Translation: Dockge/dockge
Translate-URL: https://weblate.kuma.pet/projects/dockge/dockge/pt_BR/
2023-12-08 15:49:37 +00:00
3ba267a3dc Translated using Weblate (Chinese (Traditional))
Currently translated at 100.0% (100 of 100 strings)

Translation: Dockge/dockge
Translate-URL: https://weblate.kuma.pet/projects/dockge/dockge/zh_Hant/
2023-12-08 15:49:37 +00:00
01411f2d7e Translated using Weblate (Swedish)
Currently translated at 100.0% (100 of 100 strings)

Translation: Dockge/dockge
Translate-URL: https://weblate.kuma.pet/projects/dockge/dockge/sv/
2023-12-08 15:49:37 +00:00
ba51031db6 Translated using Weblate (Russian)
Currently translated at 100.0% (100 of 100 strings)

Translation: Dockge/dockge
Translate-URL: https://weblate.kuma.pet/projects/dockge/dockge/ru/
2023-12-08 15:49:37 +00:00
daa8d12eee Translated using Weblate (German)
Currently translated at 100.0% (100 of 100 strings)

Translation: Dockge/dockge
Translate-URL: https://weblate.kuma.pet/projects/dockge/dockge/de/
2023-12-08 15:49:37 +00:00
20 changed files with 1374 additions and 512 deletions

View File

@ -6,7 +6,7 @@
A fancy, easy-to-use and reactive self-hosted docker compose.yaml stack-oriented manager. A fancy, easy-to-use and reactive self-hosted docker compose.yaml stack-oriented manager.
![GitHub Repo stars](https://img.shields.io/github/stars/louislam/dockge?logo=github) ![GitHub issues](https://img.shields.io/github/issues/louislam/dockge?logo=github) ![GitHub pull requests](https://img.shields.io/github/issues-pr/louislam/dockge?logo=github) ![Docker Pulls](https://img.shields.io/docker/pulls/louislam/dockge?logo=docker) ![Docker Image Version (latest semver)](https://img.shields.io/docker/v/louislam/dockge/latest?label=docker%20image%20ver.) ![GitHub last commit (branch)](https://img.shields.io/github/last-commit/louislam/dockge/master?logo=github) ![GitHub](https://img.shields.io/github/license/louislam/dockge?logo=github) ![GitHub Repo stars](https://img.shields.io/github/stars/louislam/dockge?logo=github) ![Docker Pulls](https://img.shields.io/docker/pulls/louislam/dockge?logo=docker) ![Docker Image Version (latest semver)](https://img.shields.io/docker/v/louislam/dockge/latest?label=docker%20image%20ver.) ![GitHub last commit (branch)](https://img.shields.io/github/last-commit/louislam/dockge/master?logo=github)
<img src="https://github.com/louislam/dockge/assets/1336778/26a583e1-ecb1-4a8d-aedf-76157d714ad7" width="900" alt="" /> <img src="https://github.com/louislam/dockge/assets/1336778/26a583e1-ecb1-4a8d-aedf-76157d714ad7" width="900" alt="" />

View File

@ -3,69 +3,55 @@ import compareVersions from "compare-versions";
import packageJSON from "../package.json"; import packageJSON from "../package.json";
import { Settings } from "./settings"; import { Settings } from "./settings";
export const obj = {
version: packageJSON.version,
latestVersion: null,
};
export default obj;
// How much time in ms to wait between update checks // How much time in ms to wait between update checks
const UPDATE_CHECKER_INTERVAL_MS = 1000 * 60 * 60 * 48; const UPDATE_CHECKER_INTERVAL_MS = 1000 * 60 * 60 * 48;
const CHECK_URL = "https://dockge.kuma.pet/version"; const CHECK_URL = "https://dockge.kuma.pet/version";
let interval : NodeJS.Timeout; class CheckVersion {
version = packageJSON.version;
latestVersion? : string;
interval? : NodeJS.Timeout;
export function startInterval() { async startInterval() {
const check = async () => { const check = async () => {
if (await Settings.get("checkUpdate") === false) { if (await Settings.get("checkUpdate") === false) {
return; return;
}
log.debug("update-checker", "Retrieving latest versions");
try {
const res = await fetch(CHECK_URL);
const data = await res.json();
// For debug
if (process.env.TEST_CHECK_VERSION === "1") {
data.slow = "1000.0.0";
} }
const checkBeta = await Settings.get("checkBeta"); log.debug("update-checker", "Retrieving latest versions");
if (checkBeta && data.beta) { try {
if (compareVersions.compare(data.beta, data.slow, ">")) { const res = await fetch(CHECK_URL);
obj.latestVersion = data.beta; const data = await res.json();
return;
// For debug
if (process.env.TEST_CHECK_VERSION === "1") {
data.slow = "1000.0.0";
} }
const checkBeta = await Settings.get("checkBeta");
if (checkBeta && data.beta) {
if (compareVersions.compare(data.beta, data.slow, ">")) {
this.latestVersion = data.beta;
return;
}
}
if (data.slow) {
this.latestVersion = data.slow;
}
} catch (_) {
log.info("update-checker", "Failed to check for new versions");
} }
if (data.slow) { };
obj.latestVersion = data.slow;
}
} catch (_) { await check();
log.info("update-checker", "Failed to check for new versions"); this.interval = setInterval(check, UPDATE_CHECKER_INTERVAL_MS);
}
};
check();
interval = setInterval(check, UPDATE_CHECKER_INTERVAL_MS);
}
/**
* Enable the check update feature
* @param value Should the check update feature be enabled?
* @returns
*/
export async function enableCheckUpdate(value : boolean) {
await Settings.set("checkUpdate", value);
clearInterval(interval);
if (value) {
startInterval();
} }
} }
const checkVersion = new CheckVersion();
export default checkVersion;

View File

@ -1,3 +0,0 @@
export class Docker {
}

View File

@ -194,6 +194,39 @@ export class DockgeServer {
// Create Socket.io // Create Socket.io
this.io = new socketIO.Server(this.httpServer, { this.io = new socketIO.Server(this.httpServer, {
cors, cors,
allowRequest: (req, callback) => {
let isOriginValid = true;
const bypass = isDev;
if (!bypass) {
let host = req.headers.host;
// If this is set, it means the request is from the browser
let origin = req.headers.origin;
// If this is from the browser, check if the origin is allowed
if (origin) {
try {
let originURL = new URL(origin);
if (host !== originURL.host) {
isOriginValid = false;
log.error("auth", `Origin (${origin}) does not match host (${host}), IP: ${req.socket.remoteAddress}`);
}
} catch (e) {
// Invalid origin url, probably not from browser
isOriginValid = false;
log.error("auth", `Invalid origin url (${origin}), IP: ${req.socket.remoteAddress}`);
}
} else {
log.info("auth", `Origin is not set, IP: ${req.socket.remoteAddress}`);
}
} else {
log.debug("auth", "Origin check is bypassed");
}
callback(null, isOriginValid);
}
}); });
this.io.on("connection", async (socket: Socket) => { this.io.on("connection", async (socket: Socket) => {
@ -308,6 +341,7 @@ export class DockgeServer {
this.sendStackList(true); this.sendStackList(true);
}); });
checkVersion.startInterval();
}); });
gracefulShutdown(this.httpServer, { gracefulShutdown(this.httpServer, {
@ -577,4 +611,35 @@ export class DockgeServer {
finalFunction() { finalFunction() {
log.info("server", "Graceful shutdown successful!"); log.info("server", "Graceful shutdown successful!");
} }
/**
* Force connected sockets of a user to refresh and disconnect.
* Used for resetting password.
* @param {string} userID
* @param {string?} currentSocketID
*/
disconnectAllSocketClients(userID: number, currentSocketID? : string) {
for (const rawSocket of this.io.sockets.sockets.values()) {
let socket = rawSocket as DockgeSocket;
if (socket.userID === userID && socket.id !== currentSocketID) {
try {
socket.emit("refresh");
socket.disconnect();
} catch (e) {
}
}
}
}
isSSL() {
return this.config.sslKey && this.config.sslCert;
}
getLocalWebSocketURL() {
const protocol = this.isSSL() ? "wss" : "ws";
const host = this.config.hostname || "localhost";
return `${protocol}://${host}:${this.config.port}`;
}
} }

View File

@ -211,6 +211,8 @@ export class MainSocketHandler extends SocketHandler {
let user = await doubleCheckPassword(socket, password.currentPassword); let user = await doubleCheckPassword(socket, password.currentPassword);
await user.resetPassword(password.newPassword); await user.resetPassword(password.newPassword);
server.disconnectAllSocketClients(user.id, socket.id);
callback({ callback({
ok: true, ok: true,
msg: "Password has been updated successfully.", msg: "Password has been updated successfully.",
@ -280,6 +282,18 @@ export class MainSocketHandler extends SocketHandler {
} }
} }
}); });
// Disconnect all other socket clients of the user
socket.on("disconnectOtherSocketClients", async () => {
try {
checkLogin(socket);
server.disconnectAllSocketClients(socket.userID, socket.id);
} catch (e) {
if (e instanceof Error) {
log.warn("disconnectOtherSocketClients", e.message);
}
}
});
} }
async login(username : string, password : string) : Promise<User | null> { async login(username : string, password : string) : Promise<User | null> {

View File

@ -1,13 +1,17 @@
/* /*
* Common utilities for backend and frontend * Common utilities for backend and frontend
*/ */
import { Document } from "yaml"; import yaml, { Document, Pair, Scalar } from "yaml";
import { DotenvParseOutput } from "dotenv";
// Init dayjs // Init dayjs
import dayjs from "dayjs"; import dayjs from "dayjs";
import timezone from "dayjs/plugin/timezone"; import timezone from "dayjs/plugin/timezone";
import utc from "dayjs/plugin/utc"; import utc from "dayjs/plugin/utc";
import relativeTime from "dayjs/plugin/relativeTime"; import relativeTime from "dayjs/plugin/relativeTime";
// @ts-ignore
import { replaceVariablesSync } from "@inventage/envsubst";
dayjs.extend(utc); dayjs.extend(utc);
dayjs.extend(timezone); dayjs.extend(timezone);
dayjs.extend(relativeTime); dayjs.extend(relativeTime);
@ -17,6 +21,11 @@ export interface LooseObject {
[key: string]: any [key: string]: any
} }
export interface BaseRes {
ok: boolean;
msg?: string;
}
let randomBytes : (numBytes: number) => Uint8Array; let randomBytes : (numBytes: number) => Uint8Array;
initRandomBytes(); initRandomBytes();
@ -340,3 +349,52 @@ export function parseDockerPort(input : string, defaultHostname : string = "loca
display: display, display: display,
}; };
} }
export function envsubst(string : string, variables : LooseObject) : string {
return replaceVariablesSync(string, variables)[0];
}
/**
* Traverse all values in the yaml and for each value, if there are template variables, replace it environment variables
* Emulates the behavior of how docker-compose handles environment variables in yaml files
* @param content Yaml string
* @param env Environment variables
* @returns string Yaml string with environment variables replaced
*/
export function envsubstYAML(content : string, env : DotenvParseOutput) : string {
const doc = yaml.parseDocument(content);
if (doc.contents) {
// @ts-ignore
for (const item of doc.contents.items) {
traverseYAML(item, env);
}
}
return doc.toString();
}
/**
* Used for envsubstYAML(...)
* @param pair
* @param env
*/
function traverseYAML(pair : Pair, env : DotenvParseOutput) : void {
// @ts-ignore
if (pair.value && pair.value.items) {
// @ts-ignore
for (const item of pair.value.items) {
if (item instanceof Pair) {
traverseYAML(item, env);
} else if (item instanceof Scalar) {
let value = item.value as unknown;
if (typeof(value) === "string") {
item.value = envsubst(value, env);
}
}
}
// @ts-ignore
} else if (pair.value && typeof(pair.value.value) === "string") {
// @ts-ignore
pair.value.value = envsubst(pair.value.value, env);
}
}

View File

@ -4,16 +4,24 @@ const input = `
`; `;
const template = ` const template = `
> [!WARNING]
>
### 🆕 New Features ### 🆕 New Features
### Improvements ### ⬆️ Improvements
### 🐞 Bug Fixes ### 🐛 Bug Fixes
### 🦎 Translation Contributions ### 🦎 Translation Contributions
### ⬆️ Security Fixes
### Others ### Others
- Other small changes, code refactoring and comment/doc updates in this repo: - Other small changes, code refactoring and comment/doc updates in this repo:
Please let me know if your username is missing, if your pull request has been merged in this version, or your commit has been included in one of the pull requests.
`; `;
const lines = input.split("\n").filter((line) => line.trim() !== ""); const lines = input.split("\n").filter((line) => line.trim() !== "");
@ -37,6 +45,12 @@ for (const line of lines) {
} }
message = message.split("* ").pop(); message = message.split("* ").pop();
console.log("-", pullRequestID, message, `(Thanks ${username})`);
let thanks = "";
if (username != "@louislam") {
thanks = `(Thanks ${username})`;
}
console.log(pullRequestID, message, thanks);
} }
console.log(template); console.log(template);

View File

@ -4,6 +4,8 @@ import readline from "readline";
import { User } from "../backend/models/user"; import { User } from "../backend/models/user";
import { DockgeServer } from "../backend/dockge-server"; import { DockgeServer } from "../backend/dockge-server";
import { log } from "../backend/log"; import { log } from "../backend/log";
import { io } from "socket.io-client";
import { BaseRes } from "../backend/util-common";
console.log("== Dockge Reset Password Tool =="); console.log("== Dockge Reset Password Tool ==");
@ -12,11 +14,10 @@ const rl = readline.createInterface({
output: process.stdout output: process.stdout
}); });
const server = new DockgeServer();
export const main = async () => { export const main = async () => {
const server = new DockgeServer();
// Check if // Check if
console.log("Connecting the database"); console.log("Connecting the database");
try { try {
await Database.init(server); await Database.init(server);
@ -47,12 +48,16 @@ export const main = async () => {
// Reset all sessions by reset jwt secret // Reset all sessions by reset jwt secret
await server.initJWTSecret(); await server.initJWTSecret();
console.log("Password reset successfully.");
// Disconnect all other socket clients of the user
await disconnectAllSocketClients(user.username, password);
break; break;
} else { } else {
console.log("Passwords do not match, please try again."); console.log("Passwords do not match, please try again.");
} }
} }
console.log("Password reset successfully.");
} }
} catch (e) { } catch (e) {
if (e instanceof Error) { if (e instanceof Error) {
@ -79,6 +84,47 @@ function question(question : string) : Promise<string> {
}); });
} }
function disconnectAllSocketClients(username : string, password : string) : Promise<void> {
return new Promise((resolve) => {
const url = server.getLocalWebSocketURL();
console.log("Connecting to " + url + " to disconnect all other socket clients");
// Disconnect all socket connections
const socket = io(url, {
transports: [ "websocket" ],
reconnection: false,
timeout: 5000,
});
socket.on("connect", () => {
socket.emit("login", {
username,
password,
}, (res : BaseRes) => {
if (res.ok) {
console.log("Logged in.");
socket.emit("disconnectOtherSocketClients");
} else {
console.warn("Login failed.");
console.warn("Please restart the server to disconnect all sessions.");
}
socket.close();
});
});
socket.on("connect_error", function () {
// The localWebSocketURL is not guaranteed to be working for some complicated Uptime Kuma setup
// Ask the user to restart the server manually
console.warn("Failed to connect to " + url);
console.warn("Please restart the server to disconnect all sessions manually.");
resolve();
});
socket.on("disconnect", () => {
resolve();
});
});
}
if (!process.env.TEST_BACKEND) { if (!process.env.TEST_BACKEND) {
main(); main();
} }

View File

@ -9,7 +9,7 @@
<div v-if="!isEditMode"> <div v-if="!isEditMode">
<span class="badge me-1" :class="bgStyle">{{ status }}</span> <span class="badge me-1" :class="bgStyle">{{ status }}</span>
<a v-for="port in service.ports" :key="port" :href="parsePort(port).url" target="_blank"> <a v-for="port in envsubstService.ports" :key="port" :href="parsePort(port).url" target="_blank">
<span class="badge me-1 bg-secondary">{{ parsePort(port).display }}</span> <span class="badge me-1 bg-secondary">{{ parsePort(port).display }}</span>
</a> </a>
</div> </div>
@ -213,16 +213,29 @@ export default defineComponent({
jsonObject() { jsonObject() {
return this.$parent.$parent.jsonConfig; return this.$parent.$parent.jsonConfig;
}, },
envsubstJSONConfig() {
return this.$parent.$parent.envsubstJSONConfig;
},
envsubstService() {
if (!this.envsubstJSONConfig.services[this.name]) {
return {};
}
return this.envsubstJSONConfig.services[this.name];
},
imageName() { imageName() {
if (this.service.image) { if (this.envsubstService.image) {
return this.service.image.split(":")[0]; return this.envsubstService.image.split(":")[0];
} else { } else {
return ""; return "";
} }
}, },
imageTag() { imageTag() {
if (this.service.image) { if (this.envsubstService.image) {
let tag = this.service.image.split(":")[1]; let tag = this.envsubstService.image.split(":")[1];
if (tag) { if (tag) {
return tag; return tag;

View File

@ -10,7 +10,7 @@
"home": "Startseite", "home": "Startseite",
"console": "Konsole", "console": "Konsole",
"registry": "Container Registry", "registry": "Container Registry",
"compose": "", "compose": "Compose",
"addFirstStackMsg": "Stelle deinen ersten Stack zusammen!", "addFirstStackMsg": "Stelle deinen ersten Stack zusammen!",
"stackName": "Stack-Name", "stackName": "Stack-Name",
"deployStack": "Deployen", "deployStack": "Deployen",
@ -72,15 +72,15 @@
"Check Update On GitHub": "Update auf GitHub überprüfen", "Check Update On GitHub": "Update auf GitHub überprüfen",
"Show update if available": "Update anzeigen, wenn verfügbar", "Show update if available": "Update anzeigen, wenn verfügbar",
"Also check beta release": "Auch Beta-Version überprüfen", "Also check beta release": "Auch Beta-Version überprüfen",
"Remember me": "Anmeldung beibehalten", "Remember me": "Angemeldet bleiben",
"Login": "Anmelden", "Login": "Anmelden",
"Username": "Benutzername", "Username": "Benutzername",
"Password": "Passwort", "Password": "Passwort",
"Settings": "Einstellungen", "Settings": "Einstellungen",
"Logout": "Abmelden", "Logout": "Abmelden",
"Lowercase only": "Nur Kleinbuchstaben", "Lowercase only": "Nur Kleinbuchstaben",
"Convert to Compose": "In Compose Syntax umwandeln", "Convert to Compose": "In Compose-Syntax umwandeln",
"Docker Run": "Docker ausführen", "Docker Run": "Docker Run",
"active": "aktiv", "active": "aktiv",
"exited": "beendet", "exited": "beendet",
"inactive": "inaktiv", "inactive": "inaktiv",

View File

@ -98,5 +98,6 @@
"reconnecting...": "Reconnecting…", "reconnecting...": "Reconnecting…",
"connecting...": "Connecting to the socket server…", "connecting...": "Connecting to the socket server…",
"url": "URL | URLs", "url": "URL | URLs",
"extra": "Extra" "extra": "Extra",
"newUpdate": "New Update"
} }

View File

@ -12,7 +12,7 @@
"registry": "Registro", "registry": "Registro",
"compose": "Compose", "compose": "Compose",
"addFirstStackMsg": "Crie sua primeira stack!", "addFirstStackMsg": "Crie sua primeira stack!",
"stackName" : "Nome da stack", "stackName": "Nome da stack",
"deployStack": "Deploy", "deployStack": "Deploy",
"deleteStack": "Excluir", "deleteStack": "Excluir",
"stopStack": "Parar", "stopStack": "Parar",
@ -22,7 +22,7 @@
"editStack": "Editar", "editStack": "Editar",
"discardStack": "Descartar", "discardStack": "Descartar",
"saveStackDraft": "Salvar", "saveStackDraft": "Salvar",
"notAvailableShort" : "N/D", "notAvailableShort": "N/D",
"deleteStackMsg": "Tem certeza que deseja excluir esta stack?", "deleteStackMsg": "Tem certeza que deseja excluir esta stack?",
"stackNotManagedByDockgeMsg": "Esta stack não é gerenciada pelo Dockge.", "stackNotManagedByDockgeMsg": "Esta stack não é gerenciada pelo Dockge.",
"primaryHostname": "Nome do Host Primário", "primaryHostname": "Nome do Host Primário",
@ -90,5 +90,13 @@
"Allowed commands:": "Comandos permitidos:", "Allowed commands:": "Comandos permitidos:",
"Internal Networks": "Redes internas", "Internal Networks": "Redes internas",
"External Networks": "Redes externas", "External Networks": "Redes externas",
"No External Networks": "Sem redes externas" "No External Networks": "Sem redes externas",
"reverseProxyMsg2": "Veja como configurar para WebSocket",
"downStack": "Parar & Encerrar",
"reverseProxyMsg1": "Utiliza proxy reverso?",
"Cannot connect to the socket server.": "Não é possível conectar ao socket server.",
"connecting...": "Conectando ao socket server…",
"url": "URL | URLs",
"extra": "Extra",
"reconnecting...": "Reconectando…"
} }

View File

@ -5,12 +5,12 @@
"PasswordsDoNotMatch": "Пароль не совпадает.", "PasswordsDoNotMatch": "Пароль не совпадает.",
"Repeat Password": "Повторите пароль", "Repeat Password": "Повторите пароль",
"Create": "Создать", "Create": "Создать",
"signedInDisp": "Авторизован как", "signedInDisp": "Авторизован как {0}",
"signedInDispDisabled": "Авторизация выключена.", "signedInDispDisabled": "Авторизация выключена.",
"home": "Главная", "home": "Главная",
"console": "Консоль", "console": "Консоль",
"registry": "Registry", "registry": "Реестр (Registry)",
"compose": "Compose", "compose": "Составить (Compose)",
"addFirstStackMsg": "Создайте свой первый стек!", "addFirstStackMsg": "Создайте свой первый стек!",
"stackName": "Имя стека", "stackName": "Имя стека",
"deployStack": "Развернуть", "deployStack": "Развернуть",
@ -90,5 +90,13 @@
"Allowed commands:": "Разрешенные команды:", "Allowed commands:": "Разрешенные команды:",
"Internal Networks": "Внутренние сети", "Internal Networks": "Внутренние сети",
"External Networks": "Внешние сети", "External Networks": "Внешние сети",
"No External Networks": "Нет внешних сетей" "No External Networks": "Нет внешних сетей",
"downStack": "Остановить и выключить",
"reverseProxyMsg1": "Использовать Реверс Прокси?",
"reconnecting...": "Переподключение…",
"Cannot connect to the socket server.": "Не удается подключиться к серверу сокетов.",
"url": "URL адрес(а)",
"extra": "Дополнительно",
"reverseProxyMsg2": "Проверьте, как настроить его для WebSocket",
"connecting...": "Подключение к серверу сокетов…"
} }

View File

@ -1,6 +1,6 @@
{ {
"languageName": "Svenska", "languageName": "Svenska",
"Create your admin account": "Skapa ditt Admin-konto.", "Create your admin account": "Skapa ditt Admin-konto",
"authIncorrectCreds": "Fel användarnamn eller lösenord.", "authIncorrectCreds": "Fel användarnamn eller lösenord.",
"PasswordsDoNotMatch": "Lösenorden matchar inte.", "PasswordsDoNotMatch": "Lösenorden matchar inte.",
"Repeat Password": "Repetera lösenord", "Repeat Password": "Repetera lösenord",
@ -12,28 +12,28 @@
"registry": "Register", "registry": "Register",
"compose": "Komponera", "compose": "Komponera",
"addFirstStackMsg": "Komponera din första stack!", "addFirstStackMsg": "Komponera din första stack!",
"stackName" : "Stacknamn", "stackName": "Stacknamn",
"deployStack": "Distribuera", "deployStack": "Distribuera",
"deleteStack": "Radera", "deleteStack": "Radera",
"stopStack": "Stop", "stopStack": "Stoppa",
"restartStack": "Starta om", "restartStack": "Starta om",
"updateStack": "Uppdatera", "updateStack": "Uppdatera",
"startStack": "Starta", "startStack": "Starta",
"downStack": "Stop & Ner", "downStack": "Stoppa & Ner",
"editStack": "Redigera", "editStack": "Redigera",
"discardStack": "Kasta", "discardStack": "Kasta",
"saveStackDraft": "Spara", "saveStackDraft": "Spara",
"notAvailableShort" : "N/A", "notAvailableShort": "N/A",
"deleteStackMsg": "Är du säker på att du vill radera stacken?", "deleteStackMsg": "Är du säker på att du vill radera stacken?",
"stackNotManagedByDockgeMsg": "Denna stacken hanteras inte av Dockge.", "stackNotManagedByDockgeMsg": "Denna stacken hanteras inte av Dockge.",
"primaryHostname": "Primärt värdnamn", "primaryHostname": "Primärt värdnamn",
"general": "Allmän", "general": "Allmän",
"container": "Container | Containrar", "container": "Container | Containrar",
"scanFolder": "Scanna Stackfolder", "scanFolder": "Skanna Stackmapp",
"dockerImage": "Bild", "dockerImage": "Avbild",
"restartPolicyUnlessStopped": "Om inte stoppas", "restartPolicyUnlessStopped": "Om inte stoppad",
"restartPolicyAlways": "Alltid", "restartPolicyAlways": "Alltid",
"restartPolicyOnFailure": "Vid Misslyckande", "restartPolicyOnFailure": "Vid misslyckande",
"restartPolicyNo": "Nej", "restartPolicyNo": "Nej",
"environmentVariable": "Miljövariabel | Miljövariabler", "environmentVariable": "Miljövariabel | Miljövariabler",
"restartPolicy": "Omstartspolicy", "restartPolicy": "Omstartspolicy",
@ -44,12 +44,12 @@
"dependsOn": "Containerberoende | Containerberoenden", "dependsOn": "Containerberoende | Containerberoenden",
"addListItem": "Lägg till {0}", "addListItem": "Lägg till {0}",
"deleteContainer": "Radera", "deleteContainer": "Radera",
"addContainer": "Lägg till Container", "addContainer": "Lägg till container",
"addNetwork": "Lägg till Nätverk", "addNetwork": "Lägg till nätverk",
"disableauth.message1": "Är du säker på att du vill <strong>inaktivera autentisering</strong>?", "disableauth.message1": "Är du säker på att du vill <strong>inaktivera autentisering</strong>?",
"disableauth.message2": "Det är designat för senarion <stong>när du ska implementera tredjeparts autentisering</strong> framör Dockge som Cloudflare Access, Authelia eller andra autentiseringsmekanismer.", "disableauth.message2": "Det är designat för scenarion <strong>där du ska implementera tredjepartsautentisering</strong> framför Dockge som Cloudflare Access, Authelia eller andra autentiseringsmekanismer.",
"passwordNotMatchMsg": "Det upprepade lösenordet matchar inte", "passwordNotMatchMsg": "Det upprepade lösenordet matchar inte.",
"autoGet": "Auto Hämta", "autoGet": "Auto-hämta",
"add": "Lägg till", "add": "Lägg till",
"Edit": "Redigera", "Edit": "Redigera",
"applyToYAML": "Lägg till i YAML", "applyToYAML": "Lägg till i YAML",
@ -57,8 +57,8 @@
"addInternalNetwork": "Lägg till", "addInternalNetwork": "Lägg till",
"Save": "Spara", "Save": "Spara",
"Language": "Språk", "Language": "Språk",
"Current User": "Nuvarande användaren", "Current User": "Nuvarande användare",
"Change Password": "Byt lösenord", "Change Password": "Ändra lösenord",
"Current Password": "Nuvarande lösenord", "Current Password": "Nuvarande lösenord",
"New Password": "Nytt lösenord", "New Password": "Nytt lösenord",
"Repeat New Password": "Upprepa nytt lösenord", "Repeat New Password": "Upprepa nytt lösenord",
@ -70,9 +70,9 @@
"I understand, please disable": "Jag förstår, vänligen inaktivera", "I understand, please disable": "Jag förstår, vänligen inaktivera",
"Leave": "Lämna", "Leave": "Lämna",
"Frontend Version": "Frontendversion", "Frontend Version": "Frontendversion",
"Check Update On GitHub": "Kontrollera Uppdatering på GitHub", "Check Update On GitHub": "Kontrollera uppdatering på GitHub",
"Show update if available": "Visa uppdatering om tillgänglig", "Show update if available": "Visa uppdatering om tillgänglig",
"Also check beta release": "Kontrollera även betaversionen", "Also check beta release": "Kontrollera även betaversioner",
"Remember me": "Kom ihåg mig", "Remember me": "Kom ihåg mig",
"Login": "Logga in", "Login": "Logga in",
"Username": "Användarnamn", "Username": "Användarnamn",
@ -80,8 +80,8 @@
"Settings": "Inställningar", "Settings": "Inställningar",
"Logout": "Logga ut", "Logout": "Logga ut",
"Lowercase only": "Endast små tecken", "Lowercase only": "Endast små tecken",
"Convert to Compose": "Omvandla till Compose", "Convert to Compose": "Omvandla till compose",
"Docker Run": "Docker Run", "Docker Run": "Docker kör",
"active": "aktiv", "active": "aktiv",
"exited": "avslutad", "exited": "avslutad",
"inactive": "inaktiv", "inactive": "inaktiv",
@ -89,7 +89,14 @@
"Security": "Säkerhet", "Security": "Säkerhet",
"About": "Om", "About": "Om",
"Allowed commands:": "Tillåtna kommandon:", "Allowed commands:": "Tillåtna kommandon:",
"Internal Networks": "Interna Nätverk", "Internal Networks": "Interna nätverk",
"External Networks": "Externa Nätverk", "External Networks": "Externa nätverk",
"No External Networks": "Inga Externa Nätverk" "No External Networks": "Inga externa nätverk",
"reverseProxyMsg1": "Används omvänd proxy?",
"connecting...": "Ansluter till socketserver…",
"Cannot connect to the socket server.": "Kan inte ansluta till socketservern.",
"reverseProxyMsg2": "Kontrollera hur man konfigurerar webbsocket",
"url": "URL | URLer",
"extra": "Extra",
"reconnecting...": "Återansluter…"
} }

View File

@ -92,7 +92,7 @@
"External Networks": "外部網路", "External Networks": "外部網路",
"No External Networks": "無外部網路", "No External Networks": "無外部網路",
"downStack": "停止", "downStack": "停止",
"reverseProxyMsg1": "在使用反向代理", "reverseProxyMsg1": "在使用反向代理",
"reverseProxyMsg2": "點擊這裡了解如何為 WebSocket 配置反向代理", "reverseProxyMsg2": "點擊這裡了解如何為 WebSocket 配置反向代理",
"Cannot connect to the socket server.": "無法連接到 Socket 伺服器。", "Cannot connect to the socket server.": "無法連接到 Socket 伺服器。",
"reconnecting...": "重新連線中…", "reconnecting...": "重新連線中…",

View File

@ -16,8 +16,8 @@
<span class="fs-4 title">Dockge</span> <span class="fs-4 title">Dockge</span>
</router-link> </router-link>
<a v-if="hasNewVersion" target="_blank" href="https://github.com/louislam/dockge/releases" class="btn btn-info me-3"> <a v-if="hasNewVersion" target="_blank" href="https://github.com/louislam/dockge/releases" class="btn btn-warning me-3">
<font-awesome-icon icon="arrow-alt-circle-up" /> {{ $t("New Update") }} <font-awesome-icon icon="arrow-alt-circle-up" /> {{ $t("newUpdate") }}
</a> </a>
<ul class="nav nav-pills"> <ul class="nav nav-pills">

View File

@ -202,6 +202,10 @@ export default defineComponent({
} }
} }
}); });
socket.on("refresh", () => {
location.reload();
});
}, },
/** /**

View File

@ -231,7 +231,7 @@ import { FontAwesomeIcon } from "@fortawesome/vue-fontawesome";
import { import {
COMBINED_TERMINAL_COLS, COMBINED_TERMINAL_COLS,
COMBINED_TERMINAL_ROWS, COMBINED_TERMINAL_ROWS,
copyYAMLComments, copyYAMLComments, envsubstYAML,
getCombinedTerminalName, getCombinedTerminalName,
getComposeTerminalName, getComposeTerminalName,
PROGRESS_TERMINAL_ROWS, PROGRESS_TERMINAL_ROWS,
@ -239,6 +239,7 @@ import {
} from "../../../backend/util-common"; } from "../../../backend/util-common";
import { BModal } from "bootstrap-vue-next"; import { BModal } from "bootstrap-vue-next";
import NetworkInput from "../components/NetworkInput.vue"; import NetworkInput from "../components/NetworkInput.vue";
import dotenv from "dotenv";
const template = `version: "3.8" const template = `version: "3.8"
services: services:
@ -277,6 +278,7 @@ export default {
return { return {
editorFocus: false, editorFocus: false,
jsonConfig: {}, jsonConfig: {},
envsubstJSONConfig: {},
yamlError: "", yamlError: "",
processing: true, processing: true,
showProgressTerminal: false, showProgressTerminal: false,
@ -297,12 +299,12 @@ export default {
computed: { computed: {
urls() { urls() {
if (!this.jsonConfig["x-dockge"] || !this.jsonConfig["x-dockge"].urls || !Array.isArray(this.jsonConfig["x-dockge"].urls)) { if (!this.envsubstJSONConfig["x-dockge"] || !this.envsubstJSONConfig["x-dockge"].urls || !Array.isArray(this.envsubstJSONConfig["x-dockge"].urls)) {
return []; return [];
} }
let urls = []; let urls = [];
for (const url of this.jsonConfig["x-dockge"].urls) { for (const url of this.envsubstJSONConfig["x-dockge"].urls) {
let display; let display;
try { try {
let obj = new URL(url); let obj = new URL(url);
@ -372,6 +374,17 @@ export default {
}, },
deep: true, deep: true,
}, },
"stack.composeENV": {
handler() {
if (this.editorFocus) {
console.debug("env code changed");
this.yamlCodeChange();
}
},
deep: true,
},
jsonConfig: { jsonConfig: {
handler() { handler() {
if (!this.editorFocus) { if (!this.editorFocus) {
@ -645,28 +658,41 @@ export default {
return highlight(code, languages.docker_env); return highlight(code, languages.docker_env);
}, },
yamlToJSON(yaml) {
let doc = parseDocument(yaml);
if (doc.errors.length > 0) {
throw doc.errors[0];
}
const config = doc.toJS() ?? {};
// Check data types
// "services" must be an object
if (!config.services) {
config.services = {};
}
if (Array.isArray(config.services) || typeof config.services !== "object") {
throw new Error("Services must be an object");
}
return {
config,
doc,
};
},
yamlCodeChange() { yamlCodeChange() {
try { try {
let doc = parseDocument(this.stack.composeYAML); let { config, doc } = this.yamlToJSON(this.stack.composeYAML);
if (doc.errors.length > 0) {
throw doc.errors[0];
}
const config = doc.toJS() ?? {};
// Check data types
// "services" must be an object
if (!config.services) {
config.services = {};
}
if (Array.isArray(config.services) || typeof config.services !== "object") {
throw new Error("Services must be an object");
}
this.yamlDoc = doc; this.yamlDoc = doc;
this.jsonConfig = config; this.jsonConfig = config;
let env = dotenv.parse(this.stack.composeENV);
let envYAML = envsubstYAML(this.stack.composeYAML, env);
this.envsubstJSONConfig = this.yamlToJSON(envYAML).config;
clearTimeout(yamlErrorTimeout); clearTimeout(yamlErrorTimeout);
this.yamlError = ""; this.yamlError = "";
} catch (e) { } catch (e) {

View File

@ -1,6 +1,6 @@
{ {
"name": "dockge", "name": "dockge",
"version": "1.3.2", "version": "1.3.3",
"type": "module", "type": "module",
"engines": { "engines": {
"node": ">= 18.0.0 && <= 18.17.1" "node": ">= 18.0.0 && <= 18.17.1"
@ -10,6 +10,7 @@
"lint": "eslint \"**/*.{ts,vue}\"", "lint": "eslint \"**/*.{ts,vue}\"",
"check-ts": "tsc --noEmit", "check-ts": "tsc --noEmit",
"start": "tsx ./backend/index.ts", "start": "tsx ./backend/index.ts",
"dev": "concurrently -k -r \"wait-on tcp:5000 && pnpm run dev:backend \" \"pnpm run dev:frontend\"",
"dev:backend": "cross-env NODE_ENV=development tsx watch --inspect ./backend/index.ts", "dev:backend": "cross-env NODE_ENV=development tsx watch --inspect ./backend/index.ts",
"dev:frontend": "cross-env NODE_ENV=development vite --host --config ./frontend/vite.config.ts", "dev:frontend": "cross-env NODE_ENV=development vite --host --config ./frontend/vite.config.ts",
"release-final": "tsx ./extra/test-docker.ts && tsx extra/update-version.ts && pnpm run build:frontend && npm run build:docker", "release-final": "tsx ./extra/test-docker.ts && tsx extra/update-version.ts && pnpm run build:frontend && npm run build:docker",
@ -24,7 +25,8 @@
"reset-password": "tsx ./extra/reset-password.ts" "reset-password": "tsx ./extra/reset-password.ts"
}, },
"dependencies": { "dependencies": {
"@homebridge/node-pty-prebuilt-multiarch": "~0.11.11", "@homebridge/node-pty-prebuilt-multiarch": "~0.11.12",
"@inventage/envsubst": "^0.16.0",
"@louislam/sqlite3": "~15.1.6", "@louislam/sqlite3": "~15.1.6",
"bcryptjs": "~2.4.3", "bcryptjs": "~2.4.3",
"check-password-strength": "~2.0.7", "check-password-strength": "~2.0.7",
@ -41,32 +43,34 @@
"jwt-decode": "~3.1.2", "jwt-decode": "~3.1.2",
"knex": "~2.5.1", "knex": "~2.5.1",
"limiter-es6-compat": "~2.1.2", "limiter-es6-compat": "~2.1.2",
"mysql2": "~3.6.3", "mysql2": "~3.6.5",
"promisify-child-process": "~4.1.2", "promisify-child-process": "~4.1.2",
"redbean-node": "~0.3.3", "redbean-node": "~0.3.3",
"socket.io": "~4.7.2", "socket.io": "~4.7.2",
"socket.io-client": "~4.7.2", "socket.io-client": "~4.7.2",
"timezones-list": "~3.0.2", "timezones-list": "~3.0.2",
"ts-command-line-args": "~2.5.1", "ts-command-line-args": "~2.5.1",
"tsx": "~3.14.0", "tsx": "~4.6.2",
"type-fest": "~4.3.3", "type-fest": "~4.3.3",
"yaml": "~2.3.4" "yaml": "~2.3.4"
}, },
"devDependencies": { "devDependencies": {
"concurrently": "^8.2.2",
"wait-on": "^7.2.0",
"@actions/github": "^6.0.0", "@actions/github": "^6.0.0",
"@fontsource/jetbrains-mono": "^5.0.17", "@fontsource/jetbrains-mono": "^5.0.18",
"@fortawesome/fontawesome-svg-core": "6.4.2", "@fortawesome/fontawesome-svg-core": "6.4.2",
"@fortawesome/free-regular-svg-icons": "6.4.2", "@fortawesome/free-regular-svg-icons": "6.4.2",
"@fortawesome/free-solid-svg-icons": "6.4.2", "@fortawesome/free-solid-svg-icons": "6.4.2",
"@fortawesome/vue-fontawesome": "3.0.3", "@fortawesome/vue-fontawesome": "3.0.3",
"@types/bcryptjs": "^2.4.6", "@types/bcryptjs": "^2.4.6",
"@types/bootstrap": "~5.2.9", "@types/bootstrap": "~5.2.10",
"@types/command-exists": "~1.2.3", "@types/command-exists": "~1.2.3",
"@types/express": "~4.17.21", "@types/express": "~4.17.21",
"@types/jsonwebtoken": "~9.0.5", "@types/jsonwebtoken": "~9.0.5",
"@typescript-eslint/eslint-plugin": "~6.8.0", "@typescript-eslint/eslint-plugin": "~6.8.0",
"@typescript-eslint/parser": "~6.8.0", "@typescript-eslint/parser": "~6.8.0",
"@vitejs/plugin-vue": "~4.5.0", "@vitejs/plugin-vue": "~4.5.2",
"bootstrap": "5.3.2", "bootstrap": "5.3.2",
"bootstrap-vue-next": "~0.14.10", "bootstrap-vue-next": "~0.14.10",
"cross-env": "~7.0.3", "cross-env": "~7.0.3",
@ -77,9 +81,9 @@
"sass": "~1.68.0", "sass": "~1.68.0",
"typescript": "~5.2.2", "typescript": "~5.2.2",
"unplugin-vue-components": "~0.25.2", "unplugin-vue-components": "~0.25.2",
"vite": "~5.0.0", "vite": "~5.0.7",
"vite-plugin-compression": "~0.5.1", "vite-plugin-compression": "~0.5.1",
"vue": "~3.3.8", "vue": "~3.3.11",
"vue-eslint-parser": "~9.3.2", "vue-eslint-parser": "~9.3.2",
"vue-i18n": "~9.5.0", "vue-i18n": "~9.5.0",
"vue-prism-editor": "2.0.0-alpha.2", "vue-prism-editor": "2.0.0-alpha.2",

1365
pnpm-lock.yaml generated

File diff suppressed because it is too large Load Diff