mirror of
https://github.com/Dokploy/dokploy
synced 2025-06-26 18:27:59 +00:00
* feat: add start monitoring remote servers * reafctor: update * refactor: update * refactor: update * refactor: update * refactor: update * refactor: update * refactor: update * refactor: * refactor: add metrics * feat: add disk monitoring * refactor: translate to english * refacotor: add stats * refactor: remove color * feat: add log server metrics * refactor: remove unused deps * refactor: add origin * refactor: add logs * refactor: update * feat: add series monitoring * refactor: add system monitoring * feat: add benchmark to optimize data * refactor: update fn * refactor: remove comments * refactor: update * refactor: exclude items * feat: add refresh rate * feat: add monitoring remote servers * refactor: update * refactor: remove unsued volumes * refactor: update monitoring * refactor: add more presets * feat: add container metrics * feat: add docker monitoring * refactor: update conversion * refactor: remove unused code * refactor: update * refactor: add docker compose logs * refactor: add docker cli * refactor: add install curl * refactor: add get update * refactor: add monitoring remote servers * refactor: add containers config * feat: add container specification * refactor: update path * refactor: add server filter * refactor: simplify logic * fix: verify if file exist before get stats * refactor: update * refactor: remove unused deps * test: add test for containers * refactor: update * refactor add memory collector * refactor: update * refactor: update * refactor: update * refactor: remove * refactor: add memory * refactor: add server memory usage * refactor: change memory * refactor: update * refactor: update * refactor: add container metrics * refactor: comment code * refactor: mount proc bind * refactor: change interval with node cron * refactor: remove opening file * refactor: use streams * refactor: remove unused ws * refactor: disable live when is all * refactor: add sqlite * refactor: update * feat: add golang benchmark * refactor: update go * refactor: update dockerfile * refactor: update db * refactor: add env * refactor: separate logic * refactor: split logic * refactor: update logs * refactor: update dockerfile * refactor: hide .env * refactor: update * chore: hide ,.ebnv * refactor: add end angle * refactor: update * refactor: update * refactor: update * refactor: update * refactor: update * refactor: update monitoring * refactor: add mount db * refactor: add metrics and url callback * refactor: add middleware * refactor: add threshold property * feat: add memory and cpu threshold notification * feat: send notifications to the server * feat: add metrics for dokploy server * refactor: add dokploy server to monitoring * refactor: update methods * refactor: add admin to useeffect * refactor: stop monitoring containers if elements are 0 * refactor: cancel request if appName is empty * refactor: reuse methods * chore; add feat monitoring * refactor: set base url * refactor: adjust monitoring * refactor: delete migrations * feat: add columns * fix: add missing flag * refactor: add free metrics * refactor: add paid monitoring * refactor: update methods * feat: improve ui * feat: add container stats * refactor: add all container metrics * refactor: add color primary * refactor: change default rate limiting refresher * refactor: update retention days * refactor: use json instead of individual properties * refactor: lint * refactor: pass json env * refactor: update * refactor: delete * refactor: update * refactor: fix types * refactor: add retention days * chore: add license * refactor: create db * refactor: update path * refactor: update setup * refactor: update * refactor: create files * refactor: update * refactor: delete * refactor: update * refactor: update token metrics * fix: typechecks * refactor: setup web server * refactor: update error handling and add monitoring * refactor: add local storage save * refactor: add spacing * refactor: update * refactor: upgrade drizzle * refactor: delete * refactor: uppgrade drizzle kit * refactor: update search with jsonB * chore: upgrade drizzle * chore: update packages * refactor: add missing type * refactor: add serverType * refactor: update url * refactor: update * refactor: update * refactor: hide monitoring on self hosted * refactor: update server * refactor: update * refactor: update * refactor: pin node version
366 lines
9.8 KiB
TypeScript
366 lines
9.8 KiB
TypeScript
import { updateServersBasedOnQuantity } from "@/pages/api/stripe/webhook";
|
|
import { createTRPCRouter, protectedProcedure } from "@/server/api/trpc";
|
|
import { db } from "@/server/db";
|
|
import {
|
|
apiCreateServer,
|
|
apiFindOneServer,
|
|
apiRemoveServer,
|
|
apiUpdateServer,
|
|
apiUpdateServerMonitoring,
|
|
applications,
|
|
compose,
|
|
mariadb,
|
|
mongo,
|
|
mysql,
|
|
postgres,
|
|
redis,
|
|
server,
|
|
} from "@/server/db/schema";
|
|
import {
|
|
IS_CLOUD,
|
|
createServer,
|
|
defaultCommand,
|
|
deleteServer,
|
|
findAdminById,
|
|
findServerById,
|
|
findServersByAdminId,
|
|
getPublicIpWithFallback,
|
|
haveActiveServices,
|
|
removeDeploymentsByServerId,
|
|
serverAudit,
|
|
serverSetup,
|
|
serverValidate,
|
|
setupMonitoring,
|
|
updateServerById,
|
|
} from "@dokploy/server";
|
|
import { TRPCError } from "@trpc/server";
|
|
import { observable } from "@trpc/server/observable";
|
|
import { and, desc, eq, getTableColumns, isNotNull, sql } from "drizzle-orm";
|
|
|
|
export const serverRouter = createTRPCRouter({
|
|
create: protectedProcedure
|
|
.input(apiCreateServer)
|
|
.mutation(async ({ ctx, input }) => {
|
|
try {
|
|
const admin = await findAdminById(ctx.user.adminId);
|
|
const servers = await findServersByAdminId(admin.adminId);
|
|
if (IS_CLOUD && servers.length >= admin.serversQuantity) {
|
|
throw new TRPCError({
|
|
code: "BAD_REQUEST",
|
|
message: "You cannot create more servers",
|
|
});
|
|
}
|
|
const project = await createServer(input, ctx.user.adminId);
|
|
return project;
|
|
} catch (error) {
|
|
throw new TRPCError({
|
|
code: "BAD_REQUEST",
|
|
message: "Error creating the server",
|
|
cause: error,
|
|
});
|
|
}
|
|
}),
|
|
|
|
one: protectedProcedure
|
|
.input(apiFindOneServer)
|
|
.query(async ({ input, ctx }) => {
|
|
const server = await findServerById(input.serverId);
|
|
if (server.adminId !== ctx.user.adminId) {
|
|
throw new TRPCError({
|
|
code: "UNAUTHORIZED",
|
|
message: "You are not authorized to access this server",
|
|
});
|
|
}
|
|
|
|
return server;
|
|
}),
|
|
getDefaultCommand: protectedProcedure
|
|
.input(apiFindOneServer)
|
|
.query(async ({ input, ctx }) => {
|
|
return defaultCommand();
|
|
}),
|
|
all: protectedProcedure.query(async ({ ctx }) => {
|
|
const result = await db
|
|
.select({
|
|
...getTableColumns(server),
|
|
totalSum: sql<number>`cast(count(${applications.applicationId}) + count(${compose.composeId}) + count(${redis.redisId}) + count(${mariadb.mariadbId}) + count(${mongo.mongoId}) + count(${mysql.mysqlId}) + count(${postgres.postgresId}) as integer)`,
|
|
})
|
|
.from(server)
|
|
.leftJoin(applications, eq(applications.serverId, server.serverId))
|
|
.leftJoin(compose, eq(compose.serverId, server.serverId))
|
|
.leftJoin(redis, eq(redis.serverId, server.serverId))
|
|
.leftJoin(mariadb, eq(mariadb.serverId, server.serverId))
|
|
.leftJoin(mongo, eq(mongo.serverId, server.serverId))
|
|
.leftJoin(mysql, eq(mysql.serverId, server.serverId))
|
|
.leftJoin(postgres, eq(postgres.serverId, server.serverId))
|
|
.where(eq(server.adminId, ctx.user.adminId))
|
|
.orderBy(desc(server.createdAt))
|
|
.groupBy(server.serverId);
|
|
|
|
return result;
|
|
}),
|
|
withSSHKey: protectedProcedure.query(async ({ ctx }) => {
|
|
const result = await db.query.server.findMany({
|
|
orderBy: desc(server.createdAt),
|
|
where: IS_CLOUD
|
|
? and(
|
|
isNotNull(server.sshKeyId),
|
|
eq(server.adminId, ctx.user.adminId),
|
|
eq(server.serverStatus, "active"),
|
|
)
|
|
: and(isNotNull(server.sshKeyId), eq(server.adminId, ctx.user.adminId)),
|
|
});
|
|
return result;
|
|
}),
|
|
setup: protectedProcedure
|
|
.input(apiFindOneServer)
|
|
.mutation(async ({ input, ctx }) => {
|
|
try {
|
|
const server = await findServerById(input.serverId);
|
|
if (server.adminId !== ctx.user.adminId) {
|
|
throw new TRPCError({
|
|
code: "UNAUTHORIZED",
|
|
message: "You are not authorized to setup this server",
|
|
});
|
|
}
|
|
const currentServer = await serverSetup(input.serverId);
|
|
return currentServer;
|
|
} catch (error) {
|
|
throw error;
|
|
}
|
|
}),
|
|
setupWithLogs: protectedProcedure
|
|
.meta({
|
|
openapi: {
|
|
path: "/deploy/server-with-logs",
|
|
method: "POST",
|
|
override: true,
|
|
enabled: false,
|
|
},
|
|
})
|
|
.input(apiFindOneServer)
|
|
.subscription(async ({ input, ctx }) => {
|
|
try {
|
|
const server = await findServerById(input.serverId);
|
|
if (server.adminId !== ctx.user.adminId) {
|
|
throw new TRPCError({
|
|
code: "UNAUTHORIZED",
|
|
message: "You are not authorized to setup this server",
|
|
});
|
|
}
|
|
return observable<string>((emit) => {
|
|
serverSetup(input.serverId, (log) => {
|
|
emit.next(log);
|
|
});
|
|
});
|
|
} catch (error) {
|
|
throw error;
|
|
}
|
|
}),
|
|
validate: protectedProcedure
|
|
.input(apiFindOneServer)
|
|
.query(async ({ input, ctx }) => {
|
|
try {
|
|
const server = await findServerById(input.serverId);
|
|
if (server.adminId !== ctx.user.adminId) {
|
|
throw new TRPCError({
|
|
code: "UNAUTHORIZED",
|
|
message: "You are not authorized to validate this server",
|
|
});
|
|
}
|
|
const response = await serverValidate(input.serverId);
|
|
return response as unknown as {
|
|
docker: {
|
|
enabled: boolean;
|
|
version: string;
|
|
};
|
|
rclone: {
|
|
enabled: boolean;
|
|
version: string;
|
|
};
|
|
nixpacks: {
|
|
enabled: boolean;
|
|
version: string;
|
|
};
|
|
buildpacks: {
|
|
enabled: boolean;
|
|
version: string;
|
|
};
|
|
isDokployNetworkInstalled: boolean;
|
|
isSwarmInstalled: boolean;
|
|
isMainDirectoryInstalled: boolean;
|
|
};
|
|
} catch (error) {
|
|
throw new TRPCError({
|
|
code: "BAD_REQUEST",
|
|
message: error instanceof Error ? error?.message : `Error: ${error}`,
|
|
cause: error as Error,
|
|
});
|
|
}
|
|
}),
|
|
|
|
security: protectedProcedure
|
|
.input(apiFindOneServer)
|
|
.query(async ({ input, ctx }) => {
|
|
try {
|
|
const server = await findServerById(input.serverId);
|
|
if (server.adminId !== ctx.user.adminId) {
|
|
throw new TRPCError({
|
|
code: "UNAUTHORIZED",
|
|
message: "You are not authorized to validate this server",
|
|
});
|
|
}
|
|
const response = await serverAudit(input.serverId);
|
|
return response as unknown as {
|
|
ufw: {
|
|
installed: boolean;
|
|
active: boolean;
|
|
defaultIncoming: string;
|
|
};
|
|
ssh: {
|
|
enabled: boolean;
|
|
keyAuth: boolean;
|
|
permitRootLogin: string;
|
|
passwordAuth: string;
|
|
usePam: string;
|
|
};
|
|
nonRootUser: {
|
|
hasValidSudoUser: boolean;
|
|
};
|
|
unattendedUpgrades: {
|
|
installed: boolean;
|
|
active: boolean;
|
|
updateEnabled: number;
|
|
upgradeEnabled: number;
|
|
};
|
|
fail2ban: {
|
|
installed: boolean;
|
|
enabled: boolean;
|
|
active: boolean;
|
|
sshEnabled: string;
|
|
sshMode: string;
|
|
};
|
|
};
|
|
} catch (error) {
|
|
throw new TRPCError({
|
|
code: "BAD_REQUEST",
|
|
message: error instanceof Error ? error?.message : `Error: ${error}`,
|
|
cause: error as Error,
|
|
});
|
|
}
|
|
}),
|
|
setupMonitoring: protectedProcedure
|
|
.input(apiUpdateServerMonitoring)
|
|
.mutation(async ({ input, ctx }) => {
|
|
try {
|
|
const server = await findServerById(input.serverId);
|
|
if (server.adminId !== ctx.user.adminId) {
|
|
throw new TRPCError({
|
|
code: "UNAUTHORIZED",
|
|
message: "You are not authorized to setup this server",
|
|
});
|
|
}
|
|
|
|
await updateServerById(input.serverId, {
|
|
metricsConfig: {
|
|
server: {
|
|
type: "Remote",
|
|
refreshRate: input.metricsConfig.server.refreshRate,
|
|
retentionDays: input.metricsConfig.server.retentionDays,
|
|
port: input.metricsConfig.server.port,
|
|
token: input.metricsConfig.server.token,
|
|
urlCallback: input.metricsConfig.server.urlCallback,
|
|
cronJob: input.metricsConfig.server.cronJob,
|
|
thresholds: {
|
|
cpu: input.metricsConfig.server.thresholds.cpu,
|
|
memory: input.metricsConfig.server.thresholds.memory,
|
|
},
|
|
},
|
|
containers: {
|
|
refreshRate: input.metricsConfig.containers.refreshRate,
|
|
services: {
|
|
include: input.metricsConfig.containers.services.include || [],
|
|
exclude: input.metricsConfig.containers.services.exclude || [],
|
|
},
|
|
},
|
|
},
|
|
});
|
|
const currentServer = await setupMonitoring(input.serverId);
|
|
return currentServer;
|
|
} catch (error) {
|
|
throw error;
|
|
}
|
|
}),
|
|
remove: protectedProcedure
|
|
.input(apiRemoveServer)
|
|
.mutation(async ({ input, ctx }) => {
|
|
try {
|
|
const server = await findServerById(input.serverId);
|
|
if (server.adminId !== ctx.user.adminId) {
|
|
throw new TRPCError({
|
|
code: "UNAUTHORIZED",
|
|
message: "You are not authorized to delete this server",
|
|
});
|
|
}
|
|
const activeServers = await haveActiveServices(input.serverId);
|
|
|
|
if (activeServers) {
|
|
throw new TRPCError({
|
|
code: "BAD_REQUEST",
|
|
message: "Server has active services, please delete them first",
|
|
});
|
|
}
|
|
const currentServer = await findServerById(input.serverId);
|
|
await removeDeploymentsByServerId(currentServer);
|
|
await deleteServer(input.serverId);
|
|
|
|
if (IS_CLOUD) {
|
|
const admin = await findAdminById(ctx.user.adminId);
|
|
|
|
await updateServersBasedOnQuantity(
|
|
admin.adminId,
|
|
admin.serversQuantity,
|
|
);
|
|
}
|
|
|
|
return currentServer;
|
|
} catch (error) {
|
|
throw error;
|
|
}
|
|
}),
|
|
update: protectedProcedure
|
|
.input(apiUpdateServer)
|
|
.mutation(async ({ input, ctx }) => {
|
|
try {
|
|
const server = await findServerById(input.serverId);
|
|
if (server.adminId !== ctx.user.adminId) {
|
|
throw new TRPCError({
|
|
code: "UNAUTHORIZED",
|
|
message: "You are not authorized to update this server",
|
|
});
|
|
}
|
|
|
|
if (server.serverStatus === "inactive") {
|
|
throw new TRPCError({
|
|
code: "NOT_FOUND",
|
|
message: "Server is inactive",
|
|
});
|
|
}
|
|
const currentServer = await updateServerById(input.serverId, {
|
|
...input,
|
|
});
|
|
|
|
return currentServer;
|
|
} catch (error) {
|
|
throw error;
|
|
}
|
|
}),
|
|
publicIp: protectedProcedure.query(async ({ ctx }) => {
|
|
if (IS_CLOUD) {
|
|
return "";
|
|
}
|
|
const ip = await getPublicIpWithFallback();
|
|
return ip;
|
|
}),
|
|
});
|