mirror of
https://github.com/Dokploy/dokploy.git
synced 2026-06-15 20:25:23 +02:00
454 lines
11 KiB
TypeScript
454 lines
11 KiB
TypeScript
import {
|
|
checkPortInUse,
|
|
createLibsql,
|
|
createMount,
|
|
deployLibsql,
|
|
findEnvironmentById,
|
|
findLibsqlById,
|
|
findProjectById,
|
|
IS_CLOUD,
|
|
rebuildDatabase,
|
|
removeLibsqlById,
|
|
removeService,
|
|
startService,
|
|
startServiceRemote,
|
|
stopService,
|
|
stopServiceRemote,
|
|
updateLibsqlById,
|
|
} from "@dokploy/server";
|
|
import {
|
|
addNewService,
|
|
checkServiceAccess,
|
|
checkServicePermissionAndAccess,
|
|
} from "@dokploy/server/services/permission";
|
|
import { TRPCError } from "@trpc/server";
|
|
import { eq } from "drizzle-orm";
|
|
import { z } from "zod";
|
|
import { createTRPCRouter, protectedProcedure } from "@/server/api/trpc";
|
|
import { audit } from "@/server/api/utils/audit";
|
|
import { db } from "@/server/db";
|
|
import {
|
|
apiChangeLibsqlStatus,
|
|
apiCreateLibsql,
|
|
apiDeployLibsql,
|
|
apiFindOneLibsql,
|
|
apiRebuildLibsql,
|
|
apiResetLibsql,
|
|
apiSaveEnvironmentVariablesLibsql,
|
|
apiSaveExternalPortsLibsql,
|
|
apiUpdateLibsql,
|
|
libsql as libsqlTable,
|
|
} from "@/server/db/schema";
|
|
export const libsqlRouter = createTRPCRouter({
|
|
create: protectedProcedure
|
|
.input(apiCreateLibsql)
|
|
.mutation(async ({ input, ctx }) => {
|
|
try {
|
|
const environment = await findEnvironmentById(input.environmentId);
|
|
const project = await findProjectById(environment.projectId);
|
|
|
|
await checkServiceAccess(ctx, project.projectId, "create");
|
|
|
|
if (IS_CLOUD && !input.serverId) {
|
|
throw new TRPCError({
|
|
code: "UNAUTHORIZED",
|
|
message: "You need to use a server to create a Libsql",
|
|
});
|
|
}
|
|
|
|
if (project.organizationId !== ctx.session.activeOrganizationId) {
|
|
throw new TRPCError({
|
|
code: "UNAUTHORIZED",
|
|
message: "You are not authorized to access this project",
|
|
});
|
|
}
|
|
const newLibsql = await createLibsql({
|
|
...input,
|
|
});
|
|
await addNewService(ctx, newLibsql.libsqlId);
|
|
|
|
await createMount({
|
|
serviceId: newLibsql.libsqlId,
|
|
serviceType: "libsql",
|
|
volumeName: `${newLibsql.appName}-data`,
|
|
mountPath: "/var/lib/sqld",
|
|
type: "volume",
|
|
});
|
|
|
|
await audit(ctx, {
|
|
action: "create",
|
|
resourceType: "service",
|
|
resourceId: newLibsql.libsqlId,
|
|
resourceName: newLibsql.appName,
|
|
});
|
|
return true;
|
|
} catch (error) {
|
|
throw error;
|
|
}
|
|
}),
|
|
one: protectedProcedure
|
|
.input(apiFindOneLibsql)
|
|
.query(async ({ input, ctx }) => {
|
|
await checkServiceAccess(ctx, input.libsqlId, "read");
|
|
|
|
const libsql = await findLibsqlById(input.libsqlId);
|
|
if (
|
|
libsql.environment.project.organizationId !==
|
|
ctx.session.activeOrganizationId
|
|
) {
|
|
throw new TRPCError({
|
|
code: "UNAUTHORIZED",
|
|
message: "You are not authorized to access this Libsql",
|
|
});
|
|
}
|
|
return libsql;
|
|
}),
|
|
|
|
start: protectedProcedure
|
|
.input(apiFindOneLibsql)
|
|
.mutation(async ({ input, ctx }) => {
|
|
await checkServicePermissionAndAccess(ctx, input.libsqlId, {
|
|
deployment: ["create"],
|
|
});
|
|
const libsql = await findLibsqlById(input.libsqlId);
|
|
|
|
if (libsql.serverId) {
|
|
await startServiceRemote(libsql.serverId, libsql.appName);
|
|
} else {
|
|
await startService(libsql.appName);
|
|
}
|
|
await updateLibsqlById(input.libsqlId, {
|
|
applicationStatus: "done",
|
|
});
|
|
|
|
await audit(ctx, {
|
|
action: "start",
|
|
resourceType: "service",
|
|
resourceId: libsql.libsqlId,
|
|
resourceName: libsql.appName,
|
|
});
|
|
return libsql;
|
|
}),
|
|
stop: protectedProcedure
|
|
.input(apiFindOneLibsql)
|
|
.mutation(async ({ input, ctx }) => {
|
|
await checkServicePermissionAndAccess(ctx, input.libsqlId, {
|
|
deployment: ["create"],
|
|
});
|
|
const libsql = await findLibsqlById(input.libsqlId);
|
|
|
|
if (libsql.serverId) {
|
|
await stopServiceRemote(libsql.serverId, libsql.appName);
|
|
} else {
|
|
await stopService(libsql.appName);
|
|
}
|
|
await updateLibsqlById(input.libsqlId, {
|
|
applicationStatus: "idle",
|
|
});
|
|
|
|
await audit(ctx, {
|
|
action: "stop",
|
|
resourceType: "service",
|
|
resourceId: libsql.libsqlId,
|
|
resourceName: libsql.appName,
|
|
});
|
|
return libsql;
|
|
}),
|
|
saveExternalPorts: protectedProcedure
|
|
.input(apiSaveExternalPortsLibsql)
|
|
.mutation(async ({ input, ctx }) => {
|
|
await checkServicePermissionAndAccess(ctx, input.libsqlId, {
|
|
service: ["create"],
|
|
});
|
|
const libsql = await findLibsqlById(input.libsqlId);
|
|
|
|
if (libsql.sqldNode === "replica" && input.externalGRPCPort !== null) {
|
|
throw new TRPCError({
|
|
code: "BAD_REQUEST",
|
|
message: "externalGRPCPort cannot be set when sqldNode is 'replica'",
|
|
});
|
|
}
|
|
|
|
const portsToCheck = [
|
|
{
|
|
port: input.externalPort,
|
|
name: "externalPort",
|
|
current: libsql.externalPort,
|
|
},
|
|
{
|
|
port: input.externalGRPCPort,
|
|
name: "externalGRPCPort",
|
|
current: libsql.externalGRPCPort,
|
|
},
|
|
{
|
|
port: input.externalAdminPort,
|
|
name: "externalAdminPort",
|
|
current: libsql.externalAdminPort,
|
|
},
|
|
];
|
|
|
|
for (const { port, name, current } of portsToCheck) {
|
|
if (port && port !== current) {
|
|
const portCheck = await checkPortInUse(
|
|
port,
|
|
libsql.serverId || undefined,
|
|
);
|
|
if (portCheck.isInUse) {
|
|
throw new TRPCError({
|
|
code: "CONFLICT",
|
|
message: `Port ${port} (${name}) is already in use by ${portCheck.conflictingContainer}`,
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
await updateLibsqlById(input.libsqlId, {
|
|
externalPort: input.externalPort,
|
|
externalGRPCPort: input.externalGRPCPort,
|
|
externalAdminPort: input.externalAdminPort,
|
|
});
|
|
await deployLibsql(input.libsqlId);
|
|
await audit(ctx, {
|
|
action: "update",
|
|
resourceType: "service",
|
|
resourceId: libsql.libsqlId,
|
|
resourceName: libsql.appName,
|
|
});
|
|
return libsql;
|
|
}),
|
|
deploy: protectedProcedure
|
|
.input(apiDeployLibsql)
|
|
.mutation(async ({ input, ctx }) => {
|
|
await checkServicePermissionAndAccess(ctx, input.libsqlId, {
|
|
deployment: ["create"],
|
|
});
|
|
const libsql = await findLibsqlById(input.libsqlId);
|
|
await audit(ctx, {
|
|
action: "deploy",
|
|
resourceType: "service",
|
|
resourceId: libsql.libsqlId,
|
|
resourceName: libsql.appName,
|
|
});
|
|
return deployLibsql(input.libsqlId);
|
|
}),
|
|
deployWithLogs: protectedProcedure
|
|
.meta({
|
|
openapi: {
|
|
path: "/deploy/libsql-with-logs",
|
|
method: "POST",
|
|
override: true,
|
|
enabled: false,
|
|
},
|
|
})
|
|
.input(apiDeployLibsql)
|
|
.subscription(async function* ({ input, ctx, signal }) {
|
|
await checkServicePermissionAndAccess(ctx, input.libsqlId, {
|
|
deployment: ["create"],
|
|
});
|
|
const queue: string[] = [];
|
|
const done = false;
|
|
|
|
deployLibsql(input.libsqlId, (log) => {
|
|
queue.push(log);
|
|
});
|
|
|
|
while (!done || queue.length > 0) {
|
|
if (queue.length > 0) {
|
|
yield queue.shift()!;
|
|
} else {
|
|
await new Promise((r) => setTimeout(r, 50));
|
|
}
|
|
|
|
if (signal?.aborted) {
|
|
return;
|
|
}
|
|
}
|
|
}),
|
|
changeStatus: protectedProcedure
|
|
.input(apiChangeLibsqlStatus)
|
|
.mutation(async ({ input, ctx }) => {
|
|
await checkServicePermissionAndAccess(ctx, input.libsqlId, {
|
|
deployment: ["create"],
|
|
});
|
|
const libsql = await findLibsqlById(input.libsqlId);
|
|
await updateLibsqlById(input.libsqlId, {
|
|
applicationStatus: input.applicationStatus,
|
|
});
|
|
await audit(ctx, {
|
|
action: "update",
|
|
resourceType: "service",
|
|
resourceId: libsql.libsqlId,
|
|
resourceName: libsql.appName,
|
|
});
|
|
return libsql;
|
|
}),
|
|
remove: protectedProcedure
|
|
.input(apiFindOneLibsql)
|
|
.mutation(async ({ input, ctx }) => {
|
|
await checkServiceAccess(ctx, input.libsqlId, "delete");
|
|
|
|
const libsql = await findLibsqlById(input.libsqlId);
|
|
|
|
if (
|
|
libsql.environment.project.organizationId !==
|
|
ctx.session.activeOrganizationId
|
|
) {
|
|
throw new TRPCError({
|
|
code: "UNAUTHORIZED",
|
|
message: "You are not authorized to delete this Libsql",
|
|
});
|
|
}
|
|
await audit(ctx, {
|
|
action: "delete",
|
|
resourceType: "service",
|
|
resourceId: libsql.libsqlId,
|
|
resourceName: libsql.appName,
|
|
});
|
|
const cleanupOperations = [
|
|
async () => await removeService(libsql?.appName, libsql.serverId),
|
|
async () => await removeLibsqlById(input.libsqlId),
|
|
];
|
|
|
|
for (const operation of cleanupOperations) {
|
|
try {
|
|
await operation();
|
|
} catch (_) {}
|
|
}
|
|
|
|
return libsql;
|
|
}),
|
|
saveEnvironment: protectedProcedure
|
|
.input(apiSaveEnvironmentVariablesLibsql)
|
|
.mutation(async ({ input, ctx }) => {
|
|
await checkServicePermissionAndAccess(ctx, input.libsqlId, {
|
|
envVars: ["write"],
|
|
});
|
|
const service = await updateLibsqlById(input.libsqlId, {
|
|
env: input.env,
|
|
});
|
|
|
|
if (!service) {
|
|
throw new TRPCError({
|
|
code: "BAD_REQUEST",
|
|
message: "Error adding environment variables",
|
|
});
|
|
}
|
|
|
|
await audit(ctx, {
|
|
action: "update",
|
|
resourceType: "service",
|
|
resourceId: input.libsqlId,
|
|
});
|
|
return true;
|
|
}),
|
|
reload: protectedProcedure
|
|
.input(apiResetLibsql)
|
|
.mutation(async ({ input, ctx }) => {
|
|
await checkServicePermissionAndAccess(ctx, input.libsqlId, {
|
|
deployment: ["create"],
|
|
});
|
|
const libsql = await findLibsqlById(input.libsqlId);
|
|
if (libsql.serverId) {
|
|
await stopServiceRemote(libsql.serverId, libsql.appName);
|
|
} else {
|
|
await stopService(libsql.appName);
|
|
}
|
|
await updateLibsqlById(input.libsqlId, {
|
|
applicationStatus: "idle",
|
|
});
|
|
|
|
if (libsql.serverId) {
|
|
await startServiceRemote(libsql.serverId, libsql.appName);
|
|
} else {
|
|
await startService(libsql.appName);
|
|
}
|
|
await updateLibsqlById(input.libsqlId, {
|
|
applicationStatus: "done",
|
|
});
|
|
await audit(ctx, {
|
|
action: "reload",
|
|
resourceType: "service",
|
|
resourceId: libsql.libsqlId,
|
|
resourceName: libsql.appName,
|
|
});
|
|
return true;
|
|
}),
|
|
update: protectedProcedure
|
|
.input(apiUpdateLibsql)
|
|
.mutation(async ({ input, ctx }) => {
|
|
const { libsqlId, ...rest } = input;
|
|
await checkServicePermissionAndAccess(ctx, libsqlId, {
|
|
service: ["create"],
|
|
});
|
|
const libsql = await updateLibsqlById(libsqlId, {
|
|
...rest,
|
|
});
|
|
|
|
if (!libsql) {
|
|
throw new TRPCError({
|
|
code: "BAD_REQUEST",
|
|
message: "Error updating Libsql",
|
|
});
|
|
}
|
|
|
|
await audit(ctx, {
|
|
action: "update",
|
|
resourceType: "service",
|
|
resourceId: libsqlId,
|
|
resourceName: libsql.appName,
|
|
});
|
|
return true;
|
|
}),
|
|
move: protectedProcedure
|
|
.input(
|
|
z.object({
|
|
libsqlId: z.string(),
|
|
targetEnvironmentId: z.string(),
|
|
}),
|
|
)
|
|
.mutation(async ({ input, ctx }) => {
|
|
await checkServicePermissionAndAccess(ctx, input.libsqlId, {
|
|
service: ["create"],
|
|
});
|
|
|
|
const updatedLibsql = await db
|
|
.update(libsqlTable)
|
|
.set({
|
|
environmentId: input.targetEnvironmentId,
|
|
})
|
|
.where(eq(libsqlTable.libsqlId, input.libsqlId))
|
|
.returning()
|
|
.then((res) => res[0]);
|
|
|
|
if (!updatedLibsql) {
|
|
throw new TRPCError({
|
|
code: "INTERNAL_SERVER_ERROR",
|
|
message: "Failed to move libsql",
|
|
});
|
|
}
|
|
|
|
await audit(ctx, {
|
|
action: "move",
|
|
resourceType: "service",
|
|
resourceId: updatedLibsql.libsqlId,
|
|
resourceName: updatedLibsql.appName,
|
|
});
|
|
return updatedLibsql;
|
|
}),
|
|
rebuild: protectedProcedure
|
|
.input(apiRebuildLibsql)
|
|
.mutation(async ({ input, ctx }) => {
|
|
await checkServicePermissionAndAccess(ctx, input.libsqlId, {
|
|
deployment: ["create"],
|
|
});
|
|
|
|
await rebuildDatabase(input.libsqlId, "libsql");
|
|
await audit(ctx, {
|
|
action: "rebuild",
|
|
resourceType: "service",
|
|
resourceId: input.libsqlId,
|
|
});
|
|
return true;
|
|
}),
|
|
});
|