2026-03-14 14:44:40 -05:00
|
|
|
import { defaultAdminPermissions, permissions, type PermissionKey } from "@mrp/shared";
|
|
|
|
|
|
|
|
|
|
import { env } from "../config/env.js";
|
|
|
|
|
import { prisma } from "./prisma.js";
|
|
|
|
|
import { hashPassword } from "./password.js";
|
|
|
|
|
import { ensureDataDirectories } from "./storage.js";
|
|
|
|
|
|
|
|
|
|
const permissionDescriptions: Record<PermissionKey, string> = {
|
|
|
|
|
[permissions.adminManage]: "Full administrative access",
|
|
|
|
|
[permissions.companyRead]: "View company settings",
|
|
|
|
|
[permissions.companyWrite]: "Update company settings",
|
|
|
|
|
[permissions.crmRead]: "View CRM records",
|
|
|
|
|
[permissions.crmWrite]: "Manage CRM records",
|
2026-03-14 21:10:35 -05:00
|
|
|
[permissions.inventoryRead]: "View inventory items and BOMs",
|
|
|
|
|
[permissions.inventoryWrite]: "Manage inventory items and BOMs",
|
2026-03-15 11:12:58 -05:00
|
|
|
[permissions.manufacturingRead]: "View manufacturing work orders and execution data",
|
|
|
|
|
[permissions.manufacturingWrite]: "Manage manufacturing work orders and execution data",
|
2026-03-14 14:44:40 -05:00
|
|
|
[permissions.filesRead]: "View attached files",
|
|
|
|
|
[permissions.filesWrite]: "Upload and manage attached files",
|
2026-03-17 23:35:37 -05:00
|
|
|
[permissions.ganttRead]: "View planning workbench",
|
2026-03-14 14:44:40 -05:00
|
|
|
[permissions.salesRead]: "View sales data",
|
2026-03-14 23:03:17 -05:00
|
|
|
[permissions.salesWrite]: "Manage quotes and sales orders",
|
2026-03-15 10:13:53 -05:00
|
|
|
[permissions.projectsRead]: "View projects and program records",
|
|
|
|
|
[permissions.projectsWrite]: "Manage projects and program records",
|
2026-03-15 00:29:41 -05:00
|
|
|
"purchasing.read": "View purchasing data",
|
|
|
|
|
"purchasing.write": "Manage purchase orders",
|
2026-03-14 14:44:40 -05:00
|
|
|
[permissions.shippingRead]: "View shipping data",
|
2026-03-14 23:48:27 -05:00
|
|
|
[permissions.shippingWrite]: "Manage shipments",
|
2026-03-14 14:44:40 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
export async function bootstrapAppData() {
|
|
|
|
|
await ensureDataDirectories();
|
|
|
|
|
|
|
|
|
|
for (const permissionKey of defaultAdminPermissions) {
|
|
|
|
|
await prisma.permission.upsert({
|
|
|
|
|
where: { key: permissionKey },
|
|
|
|
|
update: {},
|
|
|
|
|
create: {
|
|
|
|
|
key: permissionKey,
|
|
|
|
|
description: permissionDescriptions[permissionKey],
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const adminRole = await prisma.role.upsert({
|
|
|
|
|
where: { name: "Administrator" },
|
|
|
|
|
update: { description: "Full system access" },
|
|
|
|
|
create: {
|
|
|
|
|
name: "Administrator",
|
|
|
|
|
description: "Full system access",
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
const allPermissions = await prisma.permission.findMany({
|
|
|
|
|
where: {
|
|
|
|
|
key: {
|
|
|
|
|
in: defaultAdminPermissions,
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
for (const permission of allPermissions) {
|
|
|
|
|
await prisma.rolePermission.upsert({
|
|
|
|
|
where: {
|
|
|
|
|
roleId_permissionId: {
|
|
|
|
|
roleId: adminRole.id,
|
|
|
|
|
permissionId: permission.id,
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
update: {},
|
|
|
|
|
create: {
|
|
|
|
|
roleId: adminRole.id,
|
|
|
|
|
permissionId: permission.id,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const adminUser = await prisma.user.upsert({
|
|
|
|
|
where: { email: env.ADMIN_EMAIL },
|
|
|
|
|
update: {},
|
|
|
|
|
create: {
|
|
|
|
|
email: env.ADMIN_EMAIL,
|
|
|
|
|
firstName: "System",
|
|
|
|
|
lastName: "Administrator",
|
|
|
|
|
passwordHash: await hashPassword(env.ADMIN_PASSWORD),
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
await prisma.userRole.upsert({
|
|
|
|
|
where: {
|
|
|
|
|
userId_roleId: {
|
|
|
|
|
userId: adminUser.id,
|
|
|
|
|
roleId: adminRole.id,
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
update: {},
|
|
|
|
|
create: {
|
|
|
|
|
userId: adminUser.id,
|
|
|
|
|
roleId: adminRole.id,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
const existingProfile = await prisma.companyProfile.findFirst({
|
|
|
|
|
where: { isActive: true },
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (!existingProfile) {
|
|
|
|
|
await prisma.companyProfile.create({
|
|
|
|
|
data: {
|
|
|
|
|
companyName: "MRP Codex Manufacturing",
|
|
|
|
|
legalName: "MRP Codex Manufacturing LLC",
|
|
|
|
|
email: "operations@example.com",
|
|
|
|
|
phone: "+1 (555) 010-2000",
|
|
|
|
|
website: "https://example.com",
|
|
|
|
|
taxId: "99-9999999",
|
|
|
|
|
addressLine1: "100 Foundry Lane",
|
|
|
|
|
addressLine2: "Suite 200",
|
|
|
|
|
city: "Chicago",
|
|
|
|
|
state: "IL",
|
|
|
|
|
postalCode: "60601",
|
|
|
|
|
country: "USA",
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
}
|
2026-03-14 21:10:35 -05:00
|
|
|
}
|