rps-flagforge 1.0.8 ā 1.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/bin/mergeSchema.js +71 -50
- package/dist/bin/mergeSchema.js.map +1 -1
- package/package.json +1 -1
package/dist/bin/mergeSchema.js
CHANGED
|
@@ -37,44 +37,69 @@ function extractBlocks(schema, type) {
|
|
|
37
37
|
const regex = new RegExp(`${type}\\s+\\w+\\s+{[\\s\\S]*?}\\n?`, "g");
|
|
38
38
|
return schema.match(regex) || [];
|
|
39
39
|
}
|
|
40
|
-
function
|
|
41
|
-
return
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
const lines = block.split("\n").map((l) => l.trim()).filter((l) => l && !l.startsWith("//") && !l.startsWith("model") && !l.startsWith("enum") && !l.startsWith("{") && !l.startsWith("}"));
|
|
45
|
-
return lines;
|
|
40
|
+
function parseBlockFields(block) {
|
|
41
|
+
return block.split("\n").map((l) => l.trim()).filter(
|
|
42
|
+
(l) => l && !l.startsWith("//") && !l.startsWith("model") && !l.startsWith("enum") && !l.startsWith("{") && !l.startsWith("}")
|
|
43
|
+
);
|
|
46
44
|
}
|
|
47
45
|
function mergeModelFields(existing, incoming) {
|
|
48
|
-
const existingFields =
|
|
49
|
-
const incomingFields =
|
|
50
|
-
const
|
|
51
|
-
const
|
|
46
|
+
const existingFields = parseBlockFields(existing);
|
|
47
|
+
const incomingFields = parseBlockFields(incoming);
|
|
48
|
+
const mergedFields = [...existingFields];
|
|
49
|
+
const diff = [];
|
|
50
|
+
const map = new Map(existingFields.map((f) => [f.split(" ")[0], f]));
|
|
51
|
+
for (const field of incomingFields) {
|
|
52
|
+
const name = field.split(" ")[0];
|
|
53
|
+
if (!map.has(name)) {
|
|
54
|
+
mergedFields.push(field);
|
|
55
|
+
diff.push(import_chalk.default.green(`+ ${field}`));
|
|
56
|
+
} else if (map.get(name) !== field) {
|
|
57
|
+
const idx = mergedFields.findIndex((f) => f.split(" ")[0] === name);
|
|
58
|
+
mergedFields[idx] = field;
|
|
59
|
+
diff.push(import_chalk.default.yellow(`~ ${field}`));
|
|
60
|
+
}
|
|
61
|
+
}
|
|
52
62
|
const mergedBlock = existing.replace(/\{[\s\S]*\}/, `{
|
|
53
|
-
${
|
|
63
|
+
${mergedFields.join("\n ")}
|
|
54
64
|
}`);
|
|
55
|
-
return { mergedBlock,
|
|
65
|
+
return { mergedBlock, diff };
|
|
56
66
|
}
|
|
57
67
|
function mergeEnumValues(existing, incoming) {
|
|
58
|
-
const existingValues =
|
|
59
|
-
const incomingValues =
|
|
60
|
-
const
|
|
61
|
-
const
|
|
68
|
+
const existingValues = parseBlockFields(existing);
|
|
69
|
+
const incomingValues = parseBlockFields(incoming);
|
|
70
|
+
const mergedValues = [...existingValues];
|
|
71
|
+
const diff = [];
|
|
72
|
+
for (const v of incomingValues) {
|
|
73
|
+
if (!existingValues.includes(v)) {
|
|
74
|
+
mergedValues.push(v);
|
|
75
|
+
diff.push(import_chalk.default.green(`+ ${v}`));
|
|
76
|
+
}
|
|
77
|
+
}
|
|
62
78
|
const mergedBlock = existing.replace(/\{[\s\S]*\}/, `{
|
|
63
|
-
${
|
|
79
|
+
${mergedValues.join("\n ")}
|
|
64
80
|
}`);
|
|
65
|
-
return { mergedBlock,
|
|
81
|
+
return { mergedBlock, diff };
|
|
66
82
|
}
|
|
67
|
-
function
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
const
|
|
71
|
-
if (
|
|
72
|
-
|
|
73
|
-
|
|
83
|
+
function applyMergedBlocks(original, blocks, type) {
|
|
84
|
+
let content = original;
|
|
85
|
+
for (const block of blocks) {
|
|
86
|
+
const match = block.match(new RegExp(`${type}\\s+(\\w+)\\s+{`));
|
|
87
|
+
if (!match) continue;
|
|
88
|
+
const name = match[1];
|
|
89
|
+
const regex = new RegExp(`${type}\\s+${name}\\s+{[\\s\\S]*?}`, "g");
|
|
90
|
+
if (regex.test(content)) {
|
|
91
|
+
content = content.replace(regex, block);
|
|
92
|
+
} else {
|
|
93
|
+
content += `
|
|
94
|
+
|
|
95
|
+
${block}`;
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
return content;
|
|
74
99
|
}
|
|
75
100
|
async function main() {
|
|
76
|
-
console.log(import_chalk.default.blueBright.bold("\n\u26A1 FlagsForge Prisma Schema Merger\n"));
|
|
77
|
-
if (isDryRun) console.log(import_chalk.default.cyan("\u{1F50D}
|
|
101
|
+
console.log(import_chalk.default.blueBright.bold("\n\u26A1 FlagsForge Prisma Schema Smart Merger\n"));
|
|
102
|
+
if (isDryRun) console.log(import_chalk.default.cyan("\u{1F50D} Dry run: preview all changes\n"));
|
|
78
103
|
if (isForce) console.log(import_chalk.default.yellow("\u26A0\uFE0F Force mode enabled (skipping backup prompts)\n"));
|
|
79
104
|
if (!import_fs.default.existsSync(appSchema)) {
|
|
80
105
|
console.error(import_chalk.default.red("\u274C App schema not found at prisma/schema.prisma"));
|
|
@@ -86,54 +111,50 @@ async function main() {
|
|
|
86
111
|
}
|
|
87
112
|
const appContent = import_fs.default.readFileSync(appSchema, "utf-8");
|
|
88
113
|
const packageRaw = import_fs.default.readFileSync(packageSchema, "utf-8");
|
|
89
|
-
const packageContent =
|
|
114
|
+
const packageContent = packageRaw.replace(/generator\s+\w+\s+{[\s\S]*?}\n?/g, "").replace(/datasource\s+\w+\s+{[\s\S]*?}\n?/g, "");
|
|
90
115
|
const appModels = extractBlocks(appContent, "model");
|
|
91
116
|
const appEnums = extractBlocks(appContent, "enum");
|
|
92
117
|
const packageModels = extractBlocks(packageContent, "model");
|
|
93
118
|
const packageEnums = extractBlocks(packageContent, "enum");
|
|
94
|
-
const mergedModels = [
|
|
119
|
+
const mergedModels = [];
|
|
95
120
|
const modelDiffs = [];
|
|
96
121
|
for (const pModel of packageModels) {
|
|
97
122
|
const match = pModel.match(/model\s+(\w+)\s+{/);
|
|
98
123
|
if (!match) continue;
|
|
99
124
|
const name = match[1];
|
|
100
|
-
const
|
|
101
|
-
if (
|
|
102
|
-
const { mergedBlock,
|
|
103
|
-
mergedModels
|
|
104
|
-
|
|
125
|
+
const existing = appModels.find((m) => m.match(new RegExp(`model\\s+${name}\\s+{`)));
|
|
126
|
+
if (existing) {
|
|
127
|
+
const { mergedBlock, diff } = mergeModelFields(existing, pModel);
|
|
128
|
+
mergedModels.push(mergedBlock);
|
|
129
|
+
modelDiffs.push(...diff);
|
|
105
130
|
} else {
|
|
106
131
|
mergedModels.push(pModel);
|
|
107
|
-
modelDiffs.push(import_chalk.default.green(`+ ${
|
|
132
|
+
modelDiffs.push(import_chalk.default.green(`+ New model: ${name}`));
|
|
108
133
|
}
|
|
109
134
|
}
|
|
110
|
-
const mergedEnums = [
|
|
135
|
+
const mergedEnums = [];
|
|
111
136
|
const enumDiffs = [];
|
|
112
137
|
for (const pEnum of packageEnums) {
|
|
113
138
|
const match = pEnum.match(/enum\s+(\w+)\s+{/);
|
|
114
139
|
if (!match) continue;
|
|
115
140
|
const name = match[1];
|
|
116
|
-
const
|
|
117
|
-
if (
|
|
118
|
-
const { mergedBlock,
|
|
119
|
-
mergedEnums
|
|
120
|
-
|
|
141
|
+
const existing = appEnums.find((e) => e.match(new RegExp(`enum\\s+${name}\\s+{`)));
|
|
142
|
+
if (existing) {
|
|
143
|
+
const { mergedBlock, diff } = mergeEnumValues(existing, pEnum);
|
|
144
|
+
mergedEnums.push(mergedBlock);
|
|
145
|
+
enumDiffs.push(...diff);
|
|
121
146
|
} else {
|
|
122
147
|
mergedEnums.push(pEnum);
|
|
123
|
-
enumDiffs.push(import_chalk.default.green(`+ ${
|
|
148
|
+
enumDiffs.push(import_chalk.default.green(`+ New enum: ${name}`));
|
|
124
149
|
}
|
|
125
150
|
}
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
process.exit(0);
|
|
129
|
-
}
|
|
130
|
-
let mergedContent = appContent.trim();
|
|
131
|
-
mergedContent += "\n\n// ------------------------------\n// FlagsForge Schema Extensions\n// ------------------------------\n\n" + [...mergedEnums, ...mergedModels].join("\n\n");
|
|
151
|
+
let mergedContent = applyMergedBlocks(appContent, mergedModels, "model");
|
|
152
|
+
mergedContent = applyMergedBlocks(mergedContent, mergedEnums, "enum");
|
|
132
153
|
if (isDryRun) {
|
|
133
154
|
console.log(import_chalk.default.gray("----- FULL MERGE DIFF -----\n"));
|
|
134
|
-
[...enumDiffs, ...modelDiffs].forEach((
|
|
155
|
+
[...enumDiffs, ...modelDiffs].forEach((line) => console.log(line));
|
|
135
156
|
console.log(import_chalk.default.gray("\n------------------------------"));
|
|
136
|
-
console.log(import_chalk.default.cyan("\
|
|
157
|
+
console.log(import_chalk.default.cyan("\u2728 Dry run complete. No changes were made.\n"));
|
|
137
158
|
process.exit(0);
|
|
138
159
|
}
|
|
139
160
|
let shouldBackup = true;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../bin/mergeSchema.ts"],"sourcesContent":["#!/usr/bin/env node\r\nimport fs from \"fs\";\r\nimport path from \"path\";\r\nimport chalk from \"chalk\";\r\nimport prompts from \"prompts\";\r\n\r\n// -----------------------------\r\n// CLI Flags\r\n// -----------------------------\r\nconst args = process.argv.slice(2);\r\nconst isDryRun = args.includes(\"--dry-run\");\r\nconst isForce = args.includes(\"--force\");\r\n\r\n// -----------------------------\r\n// Paths\r\n// -----------------------------\r\nconst appSchema = path.resolve(process.cwd(), \"prisma/schema.prisma\");\r\nconst packageSchema = path.resolve(__dirname, \"../../prisma/schema.prisma\");\r\n\r\n// -----------------------------\r\n// Helpers\r\n// -----------------------------\r\nfunction extractBlocks(schema: string, type: \"model\" | \"enum\"): string[] {\r\n const regex = new RegExp(`${type}\\\\s+\\\\w+\\\\s+{[\\\\s\\\\S]*?}\\\\n?`, \"g\");\r\n return schema.match(regex) || [];\r\n}\r\n\r\nfunction extractNames(schema: string, type: \"model\" | \"enum\"): string[] {\r\n const regex = new RegExp(`${type}\\\\s+(\\\\w+)\\\\s+{`, \"g\");\r\n const names: string[] = [];\r\n let match;\r\n while ((match = regex.exec(schema)) !== null) {\r\n names.push(match[1]);\r\n }\r\n return names;\r\n}\r\n\r\nfunction stripNonStructuralBlocks(schema: string) {\r\n return schema\r\n .replace(/generator\\s+\\w+\\s+{[\\s\\S]*?}\\n?/g, \"\")\r\n .replace(/datasource\\s+\\w+\\s+{[\\s\\S]*?}\\n?/g, \"\");\r\n}\r\n\r\nfunction parseBlock(block: string) {\r\n const lines = block\r\n .split(\"\\n\")\r\n .map(l => l.trim())\r\n .filter(l => l && !l.startsWith(\"//\") && !l.startsWith(\"model\") && !l.startsWith(\"enum\") && !l.startsWith(\"{\") && !l.startsWith(\"}\"));\r\n return lines;\r\n}\r\n\r\nfunction mergeModelFields(existing: string, incoming: string) {\r\n const existingFields = parseBlock(existing);\r\n const incomingFields = parseBlock(incoming);\r\n const newFields = incomingFields.filter(f => !existingFields.includes(f));\r\n const mergedLines = [...existingFields, ...newFields];\r\n const mergedBlock = existing.replace(/\\{[\\s\\S]*\\}/, `{ \\n ${mergedLines.join(\"\\n \")} \\n}`);\r\n return { mergedBlock, newLines: newFields };\r\n}\r\n\r\nfunction mergeEnumValues(existing: string, incoming: string) {\r\n const existingValues = parseBlock(existing);\r\n const incomingValues = parseBlock(incoming);\r\n const newValues = incomingValues.filter(v => !existingValues.includes(v));\r\n const mergedLines = [...existingValues, ...newValues];\r\n const mergedBlock = existing.replace(/\\{[\\s\\S]*\\}/, `{ \\n ${mergedLines.join(\"\\n \")} \\n}`);\r\n return { mergedBlock, newLines: newValues };\r\n}\r\n\r\n// Colorize diff\r\nfunction diffPreview(existing: string, newLines: string[]) {\r\n const existingSet = new Set(parseBlock(existing));\r\n return existing\r\n .split(\"\\n\")\r\n .map(line => {\r\n const trimmed = line.trim();\r\n if (newLines.includes(trimmed) && !existingSet.has(trimmed)) return chalk.green(`+ ${line}`);\r\n return ` ${line}`;\r\n })\r\n .join(\"\\n\");\r\n}\r\n\r\n// -----------------------------\r\n// Main\r\n// -----------------------------\r\nasync function main() {\r\n console.log(chalk.blueBright.bold(\"\\nā” FlagsForge Prisma Schema Merger\\n\"));\r\n\r\n if (isDryRun) console.log(chalk.cyan(\"š Running in DRY RUN mode\\n\"));\r\n if (isForce) console.log(chalk.yellow(\"ā ļø Force mode enabled (skipping backup prompts)\\n\"));\r\n\r\n // Validate files\r\n if (!fs.existsSync(appSchema)) {\r\n console.error(chalk.red(\"ā App schema not found at prisma/schema.prisma\"));\r\n process.exit(1);\r\n }\r\n\r\n if (!fs.existsSync(packageSchema)) {\r\n console.error(chalk.red(\"ā Package Prisma models not found\"));\r\n process.exit(1);\r\n }\r\n\r\n const appContent = fs.readFileSync(appSchema, \"utf-8\");\r\n const packageRaw = fs.readFileSync(packageSchema, \"utf-8\");\r\n const packageContent = stripNonStructuralBlocks(packageRaw);\r\n\r\n const appModels = extractBlocks(appContent, \"model\");\r\n const appEnums = extractBlocks(appContent, \"enum\");\r\n const packageModels = extractBlocks(packageContent, \"model\");\r\n const packageEnums = extractBlocks(packageContent, \"enum\");\r\n\r\n // Merge models with diff tracking\r\n const mergedModels: string[] = [...appModels];\r\n const modelDiffs: string[] = [];\r\n for (const pModel of packageModels) {\r\n const match = pModel.match(/model\\s+(\\w+)\\s+{/);\r\n if (!match) continue;\r\n const name = match[1];\r\n const existingIndex = mergedModels.findIndex(m => m.match(new RegExp(`model\\\\s+${name}\\\\s+{`)));\r\n if (existingIndex >= 0) {\r\n const { mergedBlock, newLines } = mergeModelFields(mergedModels[existingIndex], pModel);\r\n mergedModels[existingIndex] = mergedBlock;\r\n if (newLines.length > 0) modelDiffs.push(diffPreview(mergedModels[existingIndex], newLines));\r\n } else {\r\n mergedModels.push(pModel);\r\n modelDiffs.push(chalk.green(`+ ${pModel}`));\r\n }\r\n }\r\n\r\n // Merge enums with diff tracking\r\n const mergedEnums: string[] = [...appEnums];\r\n const enumDiffs: string[] = [];\r\n for (const pEnum of packageEnums) {\r\n const match = pEnum.match(/enum\\s+(\\w+)\\s+{/);\r\n if (!match) continue;\r\n const name = match[1];\r\n const existingIndex = mergedEnums.findIndex(e => e.match(new RegExp(`enum\\\\s+${name}\\\\s+{`)));\r\n if (existingIndex >= 0) {\r\n const { mergedBlock, newLines } = mergeEnumValues(mergedEnums[existingIndex], pEnum);\r\n mergedEnums[existingIndex] = mergedBlock;\r\n if (newLines.length > 0) enumDiffs.push(diffPreview(mergedEnums[existingIndex], newLines));\r\n } else {\r\n mergedEnums.push(pEnum);\r\n enumDiffs.push(chalk.green(`+ ${pEnum}`));\r\n }\r\n }\r\n\r\n if (mergedModels.length === appModels.length && mergedEnums.length === appEnums.length) {\r\n console.log(chalk.yellow(\"ā ļø No new models or enums to merge.\\n\"));\r\n process.exit(0);\r\n }\r\n\r\n let mergedContent = appContent.trim();\r\n mergedContent +=\r\n \"\\n\\n// ------------------------------\\n\" +\r\n \"// FlagsForge Schema Extensions\\n\" +\r\n \"// ------------------------------\\n\\n\" +\r\n [...mergedEnums, ...mergedModels].join(\"\\n\\n\");\r\n\r\n // Dry-run preview\r\n if (isDryRun) {\r\n console.log(chalk.gray(\"----- FULL MERGE DIFF -----\\n\"));\r\n [...enumDiffs, ...modelDiffs].forEach(diff => console.log(diff));\r\n console.log(chalk.gray(\"\\n------------------------------\"));\r\n console.log(chalk.cyan(\"\\n⨠Dry run complete. No changes were made.\\n\"));\r\n process.exit(0);\r\n }\r\n\r\n // Backup prompt\r\n let shouldBackup = true;\r\n if (!isForce) {\r\n const response = await prompts({\r\n type: \"confirm\",\r\n name: \"backup\",\r\n message: \"Create a backup of your current schema?\",\r\n initial: true\r\n });\r\n shouldBackup = response.backup;\r\n }\r\n\r\n if (shouldBackup) {\r\n const timestamp = new Date().toISOString().replace(/[:.]/g, \"-\");\r\n const backupPath = `${appSchema}.bak-${timestamp}`;\r\n fs.copyFileSync(appSchema, backupPath);\r\n console.log(chalk.green(`ā
Backup created: ${backupPath}\\n`));\r\n }\r\n\r\n fs.writeFileSync(appSchema, mergedContent);\r\n console.log(chalk.green.bold(\"ā
Prisma schema merged successfully!\"));\r\n console.log(chalk.yellow(\"Run `npx prisma generate` to update your client.\\n\"));\r\n}\r\n\r\nmain().catch(err => {\r\n console.error(chalk.red(\"ā Error merging schemas:\"), err);\r\n process.exit(1);\r\n});\r\n"],"mappings":"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","names":["path","chalk","fs","prompts"]}
|
|
1
|
+
{"version":3,"sources":["../../bin/mergeSchema.ts"],"sourcesContent":["#!/usr/bin/env node\r\nimport fs from \"fs\";\r\nimport path from \"path\";\r\nimport chalk from \"chalk\";\r\nimport prompts from \"prompts\";\r\n\r\n// -----------------------------\r\n// CLI Flags\r\n// -----------------------------\r\nconst args = process.argv.slice(2);\r\nconst isDryRun = args.includes(\"--dry-run\");\r\nconst isForce = args.includes(\"--force\");\r\n\r\n// -----------------------------\r\n// Paths\r\n// -----------------------------\r\nconst appSchema = path.resolve(process.cwd(), \"prisma/schema.prisma\");\r\nconst packageSchema = path.resolve(__dirname, \"../../prisma/schema.prisma\");\r\n\r\n// -----------------------------\r\n// Helpers\r\n// -----------------------------\r\nfunction extractBlocks(schema: string, type: \"model\" | \"enum\"): string[] {\r\n const regex = new RegExp(`${type}\\\\s+\\\\w+\\\\s+{[\\\\s\\\\S]*?}\\\\n?`, \"g\");\r\n return schema.match(regex) || [];\r\n}\r\n\r\nfunction parseBlockFields(block: string) {\r\n return block\r\n .split(\"\\n\")\r\n .map(l => l.trim())\r\n .filter(\r\n l =>\r\n l &&\r\n !l.startsWith(\"//\") &&\r\n !l.startsWith(\"model\") &&\r\n !l.startsWith(\"enum\") &&\r\n !l.startsWith(\"{\") &&\r\n !l.startsWith(\"}\")\r\n );\r\n}\r\n\r\nfunction mergeModelFields(existing: string, incoming: string) {\r\n const existingFields = parseBlockFields(existing);\r\n const incomingFields = parseBlockFields(incoming);\r\n\r\n const mergedFields: string[] = [...existingFields];\r\n const diff: string[] = [];\r\n\r\n const map = new Map(existingFields.map(f => [f.split(\" \")[0], f]));\r\n\r\n for (const field of incomingFields) {\r\n const name = field.split(\" \")[0];\r\n if (!map.has(name)) {\r\n mergedFields.push(field);\r\n diff.push(chalk.green(`+ ${field}`));\r\n } else if (map.get(name) !== field) {\r\n const idx = mergedFields.findIndex(f => f.split(\" \")[0] === name);\r\n mergedFields[idx] = field;\r\n diff.push(chalk.yellow(`~ ${field}`));\r\n }\r\n }\r\n\r\n const mergedBlock = existing.replace(/\\{[\\s\\S]*\\}/, `{ \\n ${mergedFields.join(\"\\n \")} \\n}`);\r\n return { mergedBlock, diff };\r\n}\r\n\r\nfunction mergeEnumValues(existing: string, incoming: string) {\r\n const existingValues = parseBlockFields(existing);\r\n const incomingValues = parseBlockFields(incoming);\r\n\r\n const mergedValues = [...existingValues];\r\n const diff: string[] = [];\r\n\r\n for (const v of incomingValues) {\r\n if (!existingValues.includes(v)) {\r\n mergedValues.push(v);\r\n diff.push(chalk.green(`+ ${v}`));\r\n }\r\n }\r\n\r\n const mergedBlock = existing.replace(/\\{[\\s\\S]*\\}/, `{ \\n ${mergedValues.join(\"\\n \")} \\n}`);\r\n return { mergedBlock, diff };\r\n}\r\n\r\n// Replace blocks in schema (update existing, append only new)\r\nfunction applyMergedBlocks(original: string, blocks: string[], type: \"model\" | \"enum\") {\r\n let content = original;\r\n for (const block of blocks) {\r\n const match = block.match(new RegExp(`${type}\\\\s+(\\\\w+)\\\\s+{`));\r\n if (!match) continue;\r\n const name = match[1];\r\n\r\n const regex = new RegExp(`${type}\\\\s+${name}\\\\s+{[\\\\s\\\\S]*?}`, \"g\");\r\n if (regex.test(content)) {\r\n content = content.replace(regex, block); // update existing\r\n } else {\r\n content += `\\n\\n${block}`; // append new\r\n }\r\n }\r\n return content;\r\n}\r\n\r\n// -----------------------------\r\n// Main\r\n// -----------------------------\r\nasync function main() {\r\n console.log(chalk.blueBright.bold(\"\\nā” FlagsForge Prisma Schema Smart Merger\\n\"));\r\n\r\n if (isDryRun) console.log(chalk.cyan(\"š Dry run: preview all changes\\n\"));\r\n if (isForce) console.log(chalk.yellow(\"ā ļø Force mode enabled (skipping backup prompts)\\n\"));\r\n\r\n if (!fs.existsSync(appSchema)) {\r\n console.error(chalk.red(\"ā App schema not found at prisma/schema.prisma\"));\r\n process.exit(1);\r\n }\r\n if (!fs.existsSync(packageSchema)) {\r\n console.error(chalk.red(\"ā Package Prisma models not found\"));\r\n process.exit(1);\r\n }\r\n\r\n const appContent = fs.readFileSync(appSchema, \"utf-8\");\r\n const packageRaw = fs.readFileSync(packageSchema, \"utf-8\");\r\n const packageContent = packageRaw.replace(/generator\\s+\\w+\\s+{[\\s\\S]*?}\\n?/g, \"\").replace(/datasource\\s+\\w+\\s+{[\\s\\S]*?}\\n?/g, \"\");\r\n\r\n const appModels = extractBlocks(appContent, \"model\");\r\n const appEnums = extractBlocks(appContent, \"enum\");\r\n const packageModels = extractBlocks(packageContent, \"model\");\r\n const packageEnums = extractBlocks(packageContent, \"enum\");\r\n\r\n // Merge models\r\n const mergedModels: string[] = [];\r\n const modelDiffs: string[] = [];\r\n for (const pModel of packageModels) {\r\n const match = pModel.match(/model\\s+(\\w+)\\s+{/);\r\n if (!match) continue;\r\n const name = match[1];\r\n\r\n const existing = appModels.find(m => m.match(new RegExp(`model\\\\s+${name}\\\\s+{`)));\r\n if (existing) {\r\n const { mergedBlock, diff } = mergeModelFields(existing, pModel);\r\n mergedModels.push(mergedBlock);\r\n modelDiffs.push(...diff);\r\n } else {\r\n mergedModels.push(pModel);\r\n modelDiffs.push(chalk.green(`+ New model: ${name}`));\r\n }\r\n }\r\n\r\n // Merge enums\r\n const mergedEnums: string[] = [];\r\n const enumDiffs: string[] = [];\r\n for (const pEnum of packageEnums) {\r\n const match = pEnum.match(/enum\\s+(\\w+)\\s+{/);\r\n if (!match) continue;\r\n const name = match[1];\r\n\r\n const existing = appEnums.find(e => e.match(new RegExp(`enum\\\\s+${name}\\\\s+{`)));\r\n if (existing) {\r\n const { mergedBlock, diff } = mergeEnumValues(existing, pEnum);\r\n mergedEnums.push(mergedBlock);\r\n enumDiffs.push(...diff);\r\n } else {\r\n mergedEnums.push(pEnum);\r\n enumDiffs.push(chalk.green(`+ New enum: ${name}`));\r\n }\r\n }\r\n\r\n // Apply merged blocks in-place\r\n let mergedContent = applyMergedBlocks(appContent, mergedModels, \"model\");\r\n mergedContent = applyMergedBlocks(mergedContent, mergedEnums, \"enum\");\r\n\r\n // Dry-run preview\r\n if (isDryRun) {\r\n console.log(chalk.gray(\"----- FULL MERGE DIFF -----\\n\"));\r\n [...enumDiffs, ...modelDiffs].forEach(line => console.log(line));\r\n console.log(chalk.gray(\"\\n------------------------------\"));\r\n console.log(chalk.cyan(\"⨠Dry run complete. No changes were made.\\n\"));\r\n process.exit(0);\r\n }\r\n\r\n // Backup prompt\r\n let shouldBackup = true;\r\n if (!isForce) {\r\n const response = await prompts({\r\n type: \"confirm\",\r\n name: \"backup\",\r\n message: \"Create a backup of your current schema?\",\r\n initial: true,\r\n });\r\n shouldBackup = response.backup;\r\n }\r\n\r\n if (shouldBackup) {\r\n const timestamp = new Date().toISOString().replace(/[:.]/g, \"-\");\r\n const backupPath = `${appSchema}.bak-${timestamp}`;\r\n fs.copyFileSync(appSchema, backupPath);\r\n console.log(chalk.green(`ā
Backup created: ${backupPath}\\n`));\r\n }\r\n\r\n fs.writeFileSync(appSchema, mergedContent);\r\n console.log(chalk.green.bold(\"ā
Prisma schema merged successfully!\"));\r\n console.log(chalk.yellow(\"Run `npx prisma generate` to update your client.\\n\"));\r\n}\r\n\r\nmain().catch(err => {\r\n console.error(chalk.red(\"ā Error merging schemas:\"), err);\r\n process.exit(1);\r\n});\r\n"],"mappings":"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","names":["path","chalk","fs","prompts"]}
|