prisma-generator-plantuml-erd 1.0.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/CHANGELOG.md ADDED
@@ -0,0 +1,18 @@
1
+ # 1.0.0 (2022-11-08)
2
+
3
+
4
+ ### Bug Fixes
5
+
6
+ * divider line ([0bca83e](https://github.com/dbgso/prisma-generator-plantuml-erd/commit/0bca83e0cdfc0a28586b231a122e76a4070e6291))
7
+ * enum relation ([883277b](https://github.com/dbgso/prisma-generator-plantuml-erd/commit/883277bb13a8125d4165ddd53ae56f449542d912))
8
+ * npm audit ([17a5b91](https://github.com/dbgso/prisma-generator-plantuml-erd/commit/17a5b91d87fcc042b4031722e7d023ec8bdcbe15))
9
+ * option type definition ([abfb012](https://github.com/dbgso/prisma-generator-plantuml-erd/commit/abfb0125633e789c98535bef242585b985d2e900))
10
+ * prisma boolean config ([03da82e](https://github.com/dbgso/prisma-generator-plantuml-erd/commit/03da82e052cbd7b495a26b5566c5b35c73e3b027))
11
+ * safe write file ([0e5166f](https://github.com/dbgso/prisma-generator-plantuml-erd/commit/0e5166f12925da0d376a34f4d51db55cf9c915f9))
12
+
13
+
14
+ ### Features
15
+
16
+ * add option to export uml per tables ([31cc2ad](https://github.com/dbgso/prisma-generator-plantuml-erd/commit/31cc2adc87689650f9889d847231fa17ee867316))
17
+ * enum relations ([64a9b6c](https://github.com/dbgso/prisma-generator-plantuml-erd/commit/64a9b6cab4db1a0cdcb5e4d862aab2ff593b3a19))
18
+ * implements prototyping ([c20f7b0](https://github.com/dbgso/prisma-generator-plantuml-erd/commit/c20f7b0683c2d387f0f0870623859fe4286aef72))
package/dist/bin.js ADDED
@@ -0,0 +1,5 @@
1
+ #!/usr/bin/env node
2
+ "use strict";
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ require("./generator");
5
+ //# sourceMappingURL=bin.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"bin.js","sourceRoot":"","sources":["../src/bin.ts"],"names":[],"mappings":";;;AACA,uBAAoB"}
@@ -0,0 +1,5 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.GENERATOR_NAME = void 0;
4
+ exports.GENERATOR_NAME = 'prisma-generator-plantuml-erd';
5
+ //# sourceMappingURL=constants.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"constants.js","sourceRoot":"","sources":["../src/constants.ts"],"names":[],"mappings":";;;AAAa,QAAA,cAAc,GAAG,+BAA+B,CAAA"}
@@ -0,0 +1,24 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const generator_helper_1 = require("@prisma/generator-helper");
4
+ const constants_1 = require("./constants");
5
+ const plantuml_erd_1 = require("./plantuml-erd");
6
+ const { version } = require('../package.json');
7
+ (0, generator_helper_1.generatorHandler)({
8
+ onManifest() {
9
+ return {
10
+ version,
11
+ defaultOutput: '../generated',
12
+ prettyName: constants_1.GENERATOR_NAME,
13
+ };
14
+ },
15
+ onGenerate: async (options) => {
16
+ var _a;
17
+ const generator = new plantuml_erd_1.PlantUmlErdGenerator({
18
+ output: (_a = options.generator.output) === null || _a === void 0 ? void 0 : _a.value,
19
+ ...options.generator.config,
20
+ });
21
+ await generator.generate(options.dmmf);
22
+ },
23
+ });
24
+ //# sourceMappingURL=generator.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"generator.js","sourceRoot":"","sources":["../src/generator.ts"],"names":[],"mappings":";;AAAA,+DAA8E;AAC9E,2CAA6C;AAC7C,iDAAsD;AAEtD,MAAM,EAAE,OAAO,EAAE,GAAG,OAAO,CAAC,iBAAiB,CAAC,CAAC;AAE/C,IAAA,mCAAgB,EAAC;IACf,UAAU;QACR,OAAO;YACL,OAAO;YACP,aAAa,EAAE,cAAc;YAC7B,UAAU,EAAE,0BAAc;SAC3B,CAAC;IACJ,CAAC;IACD,UAAU,EAAE,KAAK,EAAE,OAAyB,EAAE,EAAE;;QAC9C,MAAM,SAAS,GAAG,IAAI,mCAAoB,CAAC;YACzC,MAAM,EAAE,MAAA,OAAO,CAAC,SAAS,CAAC,MAAM,0CAAE,KAAK;YACvC,GAAG,OAAO,CAAC,SAAS,CAAC,MAAM;SAC5B,CAAC,CAAC;QACH,MAAM,SAAS,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;IACzC,CAAC;CACF,CAAC,CAAC"}
@@ -0,0 +1,9 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.genEnum = void 0;
4
+ const genEnum = ({ name, values }) => {
5
+ const enumValues = values.map(({ name }) => `${name}="${name}"`).join(',\n');
6
+ return `enum ${name} { \n${enumValues}\n }`;
7
+ };
8
+ exports.genEnum = genEnum;
9
+ //# sourceMappingURL=genEnum.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"genEnum.js","sourceRoot":"","sources":["../../src/helpers/genEnum.ts"],"names":[],"mappings":";;;AAEO,MAAM,OAAO,GAAG,CAAC,EAAE,IAAI,EAAE,MAAM,EAAsB,EAAE,EAAE;IAC9D,MAAM,UAAU,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,EAAE,EAAE,EAAE,CAAC,GAAG,IAAI,KAAK,IAAI,GAAG,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;IAE5E,OAAO,QAAQ,IAAI,QAAQ,UAAU,MAAM,CAAA;AAC7C,CAAC,CAAA;AAJY,QAAA,OAAO,WAInB"}
@@ -0,0 +1,11 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.getDMMFFromFile = void 0;
4
+ const sdk_1 = require("@prisma/sdk");
5
+ const getDMMFFromFile = (filepath) => {
6
+ return (0, sdk_1.getDMMF)({
7
+ datamodel: (0, sdk_1.getSchemaSync)(filepath),
8
+ });
9
+ };
10
+ exports.getDMMFFromFile = getDMMFFromFile;
11
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/helpers/index.ts"],"names":[],"mappings":";;;AAAA,qCAAqD;AAE9C,MAAM,eAAe,GAAG,CAAC,QAAgB,EAAE,EAAE;IAClD,OAAO,IAAA,aAAO,EAAC;QACb,SAAS,EAAE,IAAA,mBAAa,EAAC,QAAQ,CAAC;KACnC,CAAC,CAAC;AACL,CAAC,CAAC;AAJW,QAAA,eAAe,mBAI1B"}
@@ -0,0 +1,229 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.PlantUmlErdGenerator = void 0;
7
+ const fs_1 = require("fs");
8
+ const path_1 = __importDefault(require("path"));
9
+ const types_1 = require("./types");
10
+ class PlantUmlErdGenerator {
11
+ constructor(config) {
12
+ this.config = types_1.PlantUmlErdGeneratorConfigsSchema.parse(config);
13
+ }
14
+ filter(dmmf, targetTableName) {
15
+ if (!targetTableName)
16
+ return dmmf;
17
+ const target = dmmf.datamodel.models.find((e) => e.name === targetTableName);
18
+ if (!target)
19
+ return dmmf;
20
+ const enums = target.fields
21
+ .filter((f) => f.kind === 'enum')
22
+ .map((f) => f.type);
23
+ const models = target.fields
24
+ .filter((f) => f.kind === 'object')
25
+ .map((f) => f.type);
26
+ const cloned = JSON.parse(JSON.stringify(dmmf));
27
+ cloned.datamodel.enums = cloned.datamodel.enums.filter((e) => enums.includes(e.name));
28
+ cloned.datamodel.models = cloned.datamodel.models.filter((e) => models.includes(e.name) || e.name === targetTableName);
29
+ return cloned;
30
+ }
31
+ async generate(dmmf) {
32
+ if (this.config.debug) {
33
+ await fs_1.promises.writeFile('/tmp/example.json', JSON.stringify(dmmf, null, 2));
34
+ }
35
+ const results = this._generate(dmmf, 'erd');
36
+ if (this.config.exportPerTables) {
37
+ results.push(...this.dumpSubRelations(dmmf));
38
+ }
39
+ fs_1.promises.mkdir(path_1.default.dirname(this.config.output), {
40
+ recursive: true,
41
+ });
42
+ await fs_1.promises.writeFile(this.config.output, results.join('\n'));
43
+ }
44
+ dumpSubRelations(dmmf) {
45
+ const results = [];
46
+ for (const model of dmmf.datamodel.models) {
47
+ const filteredDmmf = this.filter(dmmf, model.name);
48
+ results.push(...this._generate(filteredDmmf, model.name));
49
+ }
50
+ return results;
51
+ }
52
+ _generate(dmmf, diagramName) {
53
+ const results = [`@startuml ${diagramName}`, 'skinparam linetype ortho'];
54
+ results.push(...this.drawEnums(dmmf));
55
+ results.push(...this.drawEntities(dmmf));
56
+ results.push(...this.drawRelations(dmmf));
57
+ results.push(...this.drawEnumRelations(dmmf));
58
+ results.push('@enduml');
59
+ return results;
60
+ }
61
+ drawEnumRelations(dmmf) {
62
+ const results = [];
63
+ results.push(`' enum relations`);
64
+ for (const model of dmmf.datamodel.models) {
65
+ for (const enumRelation of model.fields.filter((f) => f.kind === 'enum')) {
66
+ const isExists = dmmf.datamodel.enums.some((e) => e.name === enumRelation.type);
67
+ if (!isExists)
68
+ continue;
69
+ const lines = [];
70
+ lines.push(model.name);
71
+ lines.push(' ');
72
+ lines.push('|o');
73
+ lines.push(this.config.lineLength);
74
+ lines.push(enumRelation.isList ? '|{' : '||');
75
+ lines.push(' ');
76
+ lines.push(enumRelation.type);
77
+ results.push(lines.join(''));
78
+ }
79
+ }
80
+ return results;
81
+ }
82
+ drawEnums(dmmf) {
83
+ const results = [];
84
+ for (const e of dmmf.datamodel.enums) {
85
+ const enumDefineline = ['enum "'];
86
+ enumDefineline.push(e.name);
87
+ if (e.documentation) {
88
+ enumDefineline.push(`\\n${e.documentation}`);
89
+ }
90
+ enumDefineline.push(`" as ${e.name} {`);
91
+ results.push(enumDefineline.join(''));
92
+ for (const value of e.values) {
93
+ const enumValueLine = [' ', value.name];
94
+ if (value.dbName) {
95
+ enumValueLine.push(': ');
96
+ enumValueLine.push(value.dbName);
97
+ }
98
+ results.push(enumValueLine.join(''));
99
+ }
100
+ results.push(`}`);
101
+ }
102
+ return results;
103
+ }
104
+ drawEntities(dmmf) {
105
+ const results = [];
106
+ for (const model of dmmf.datamodel.models) {
107
+ const idField = model.fields.find((f) => f.isId);
108
+ const name = this.config.usePhysicalTableName ? model.dbName : model.name;
109
+ const documentation = model.documentation
110
+ ? `\\n${model.documentation}`
111
+ : '';
112
+ results.push(`entity "${name}${documentation}" as ${model.name} {`);
113
+ results.push(`* ${idField === null || idField === void 0 ? void 0 : idField.name} [PK] : ${idField === null || idField === void 0 ? void 0 : idField.type} ${(idField === null || idField === void 0 ? void 0 : idField.documentation) || ''}`);
114
+ results.push('--');
115
+ for (const field of model.fields.filter((f) => !f.isId)) {
116
+ const line = this._buildField(field, model.fields);
117
+ if (line) {
118
+ results.push(` ${line}`);
119
+ }
120
+ }
121
+ results.push(`}`);
122
+ results.push(``);
123
+ }
124
+ return results;
125
+ }
126
+ drawRelations(dmmf) {
127
+ const results = [];
128
+ const manyToManyList = this._findManyToMany(dmmf.datamodel.models);
129
+ results.push(`' Relations`);
130
+ for (const model of dmmf.datamodel.models) {
131
+ for (const field of model.fields) {
132
+ const isExists = dmmf.datamodel.models.some((model) => model.name === field.type);
133
+ if (!isExists)
134
+ continue;
135
+ const toMany = field.isList;
136
+ const relationName = field.relationName;
137
+ if (relationName && manyToManyList.includes(relationName)) {
138
+ continue;
139
+ }
140
+ for (const _ of field.relationFromFields || []) {
141
+ results.push(`${model.name} ${this._buildRelationLineFromOne(field, model.fields)} ${field.type}`);
142
+ }
143
+ }
144
+ }
145
+ results.push(`' ManyToMany Relations`);
146
+ for (const manyToManyRelationName of manyToManyList) {
147
+ const model = dmmf.datamodel.models
148
+ .find((model) => model.fields.find((field) => field.relationName === manyToManyRelationName));
149
+ if (!model)
150
+ continue;
151
+ const field = model.fields.find((field) => field.relationName === manyToManyRelationName);
152
+ if (!field)
153
+ continue;
154
+ results.push(`${model === null || model === void 0 ? void 0 : model.name} }o${this.config.lineLength}o{ ${field.type}`);
155
+ }
156
+ return results;
157
+ }
158
+ _findManyToMany(models) {
159
+ var _a;
160
+ const set = new Set();
161
+ const duplicated = [];
162
+ for (const model of models) {
163
+ for (const field of model.fields) {
164
+ const relationName = field.relationName;
165
+ if (!relationName)
166
+ continue;
167
+ if ((_a = field.relationFromFields) === null || _a === void 0 ? void 0 : _a.length)
168
+ continue;
169
+ if (set.has(relationName)) {
170
+ duplicated.push(relationName);
171
+ }
172
+ set.add(relationName);
173
+ }
174
+ }
175
+ return duplicated;
176
+ }
177
+ _buildRelationLineFromOne(field, fields) {
178
+ var _a;
179
+ const strings = [];
180
+ if (!((_a = field.relationFromFields) === null || _a === void 0 ? void 0 : _a.length)) {
181
+ throw new Error('invalid field');
182
+ }
183
+ const relationFromField = field.relationFromFields[0];
184
+ const foreignKeyField = fields.find((field) => field.name === relationFromField);
185
+ if (!foreignKeyField)
186
+ throw new Error('foreign key is not exists');
187
+ if (foreignKeyField.isUnique) {
188
+ strings.push('|');
189
+ }
190
+ else {
191
+ strings.push('}');
192
+ }
193
+ strings.push(this.config.relationMiniumOne ? '|' : 'o');
194
+ strings.push(this.config.lineLength);
195
+ if (foreignKeyField.isRequired) {
196
+ strings.push('||');
197
+ }
198
+ else {
199
+ strings.push('o|');
200
+ }
201
+ return strings.join('');
202
+ }
203
+ _buildFieldDocument(field, fields) {
204
+ const result = [`${field.name}`];
205
+ if (field.documentation && !field.documentation.includes('@')) {
206
+ result.push(`(${field.documentation})`);
207
+ }
208
+ result.push(':');
209
+ const foreignKey = fields.find((f) => { var _a; return (_a = f.relationFromFields) === null || _a === void 0 ? void 0 : _a.includes(field.name); });
210
+ if (foreignKey) {
211
+ result.push('[FK]');
212
+ result.push(foreignKey.type);
213
+ }
214
+ else {
215
+ result.push(field.type);
216
+ }
217
+ return result.join(' ');
218
+ }
219
+ _buildField(field, fields) {
220
+ if (field.relationName)
221
+ return undefined;
222
+ const name = this._buildFieldDocument(field, fields);
223
+ if (field.isRequired)
224
+ return `**${name}**`;
225
+ return name;
226
+ }
227
+ }
228
+ exports.PlantUmlErdGenerator = PlantUmlErdGenerator;
229
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/plantuml-erd/index.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,18 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.PlantUmlErdGeneratorConfigsSchema = void 0;
4
+ const zod_1 = require("zod");
5
+ const optionalStrinToBooleanSchema = (flg) => zod_1.z
6
+ .enum(['true', 'false'])
7
+ .optional()
8
+ .default(flg ? 'true' : 'false')
9
+ .transform((args) => args === 'true');
10
+ exports.PlantUmlErdGeneratorConfigsSchema = zod_1.z.object({
11
+ output: zod_1.z.string().optional().default('./erd.puml'),
12
+ usePhysicalTableName: optionalStrinToBooleanSchema(false),
13
+ lineLength: zod_1.z.string().regex(/-+/).optional().default('--'),
14
+ relationMiniumOne: optionalStrinToBooleanSchema(false),
15
+ debug: optionalStrinToBooleanSchema(false),
16
+ exportPerTables: optionalStrinToBooleanSchema(false),
17
+ });
18
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/plantuml-erd/types/index.ts"],"names":[],"mappings":";;;AAAA,6BAAwB;AACxB,MAAM,4BAA4B,GAAG,CAAC,GAAY,EAAE,EAAE,CACpD,OAAC;KACE,IAAI,CAAC,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;KACvB,QAAQ,EAAE;KACV,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,OAAO,CAAC;KAC/B,SAAS,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,KAAK,MAAM,CAAC,CAAC;AAE7B,QAAA,iCAAiC,GAAG,OAAC,CAAC,MAAM,CAAC;IACxD,MAAM,EAAE,OAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,EAAE,CAAC,OAAO,CAAC,YAAY,CAAC;IACnD,oBAAoB,EAAE,4BAA4B,CAAC,KAAK,CAAC;IACzD,UAAU,EAAE,OAAC,CAAC,MAAM,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,QAAQ,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC;IAC3D,iBAAiB,EAAE,4BAA4B,CAAC,KAAK,CAAC;IACtD,KAAK,EAAE,4BAA4B,CAAC,KAAK,CAAC;IAC1C,eAAe,EAAE,4BAA4B,CAAC,KAAK,CAAC;CACrD,CAAC,CAAC"}
@@ -0,0 +1,26 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.formatFile = void 0;
7
+ const prettier_1 = __importDefault(require("prettier"));
8
+ const formatFile = (content) => {
9
+ return new Promise((res, rej) => prettier_1.default.resolveConfig(process.cwd()).then((options) => {
10
+ if (!options) {
11
+ res(content);
12
+ }
13
+ try {
14
+ const formatted = prettier_1.default.format(content, {
15
+ ...options,
16
+ parser: 'typescript',
17
+ });
18
+ res(formatted);
19
+ }
20
+ catch (error) {
21
+ rej(error);
22
+ }
23
+ }));
24
+ };
25
+ exports.formatFile = formatFile;
26
+ //# sourceMappingURL=formatFile.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"formatFile.js","sourceRoot":"","sources":["../../src/utils/formatFile.ts"],"names":[],"mappings":";;;;;;AAAA,wDAA+B;AAExB,MAAM,UAAU,GAAG,CAAC,OAAe,EAAmB,EAAE;IAC7D,OAAO,IAAI,OAAO,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE,CAC9B,kBAAQ,CAAC,aAAa,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,OAAO,EAAE,EAAE;QACrD,IAAI,CAAC,OAAO,EAAE;YACZ,GAAG,CAAC,OAAO,CAAC,CAAA;SACb;QAED,IAAI;YACF,MAAM,SAAS,GAAG,kBAAQ,CAAC,MAAM,CAAC,OAAO,EAAE;gBACzC,GAAG,OAAO;gBACV,MAAM,EAAE,YAAY;aACrB,CAAC,CAAA;YAEF,GAAG,CAAC,SAAS,CAAC,CAAA;SACf;QAAC,OAAO,KAAK,EAAE;YACd,GAAG,CAAC,KAAK,CAAC,CAAA;SACX;IACH,CAAC,CAAC,CACH,CAAA;AACH,CAAC,CAAA;AAnBY,QAAA,UAAU,cAmBtB"}
@@ -0,0 +1,17 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.writeFileSafely = void 0;
7
+ const fs_1 = __importDefault(require("fs"));
8
+ const path_1 = __importDefault(require("path"));
9
+ const formatFile_1 = require("./formatFile");
10
+ const writeFileSafely = async (writeLocation, content) => {
11
+ fs_1.default.mkdirSync(path_1.default.dirname(writeLocation), {
12
+ recursive: true,
13
+ });
14
+ fs_1.default.writeFileSync(writeLocation, await (0, formatFile_1.formatFile)(content));
15
+ };
16
+ exports.writeFileSafely = writeFileSafely;
17
+ //# sourceMappingURL=writeFileSafely.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"writeFileSafely.js","sourceRoot":"","sources":["../../src/utils/writeFileSafely.ts"],"names":[],"mappings":";;;;;;AAAA,4CAAmB;AACnB,gDAAuB;AACvB,6CAAyC;AAElC,MAAM,eAAe,GAAG,KAAK,EAAE,aAAqB,EAAE,OAAY,EAAE,EAAE;IAC3E,YAAE,CAAC,SAAS,CAAC,cAAI,CAAC,OAAO,CAAC,aAAa,CAAC,EAAE;QACxC,SAAS,EAAE,IAAI;KAChB,CAAC,CAAA;IAEF,YAAE,CAAC,aAAa,CAAC,aAAa,EAAE,MAAM,IAAA,uBAAU,EAAC,OAAO,CAAC,CAAC,CAAA;AAC5D,CAAC,CAAA;AANY,QAAA,eAAe,mBAM3B"}
package/package.json ADDED
@@ -0,0 +1,83 @@
1
+ {
2
+ "name": "prisma-generator-plantuml-erd",
3
+ "description": "Provide a description about your generator",
4
+ "version": "1.0.0",
5
+ "main": "dist/generator.js",
6
+ "license": "MIT",
7
+ "bin": {
8
+ "prisma-generator-plantuml-erd": "dist/bin.js"
9
+ },
10
+ "engines": {
11
+ "node": ">=14.0"
12
+ },
13
+ "scripts": {
14
+ "start": "node dist/bin.js",
15
+ "dev": "npx tsc -w",
16
+ "build": "npx tsc",
17
+ "prepack": "yarn build",
18
+ "test": "jest"
19
+ },
20
+ "dependencies": {
21
+ "@prisma/client": "3.12.0",
22
+ "@prisma/generator-helper": "3.12.0",
23
+ "zod": "3.19.1"
24
+ },
25
+ "peerDependencies": {
26
+ "@prisma/client": "3.12.0",
27
+ "@prisma/generator-helper": "3.12.0",
28
+ "zod": "3.19.1"
29
+ },
30
+ "devDependencies": {
31
+ "@prisma/sdk": "^3.15.2",
32
+ "@semantic-release/changelog": "^6.0.1",
33
+ "@semantic-release/git": "^10.0.1",
34
+ "@types/jest": "^29.2.2",
35
+ "@types/node": "^17.0.21",
36
+ "@types/prettier": "^2.4.2",
37
+ "jest": "^29.3.0",
38
+ "prettier": "^2.7.1",
39
+ "prisma": "^3.12.0",
40
+ "semantic-release": "^19.0.5",
41
+ "ts-jest": "^29.0.3",
42
+ "typescript": "4.6.2"
43
+ },
44
+ "homepage": "Link to homepage or github readme here",
45
+ "repository": {
46
+ "type": "git",
47
+ "url": "https://github.com/dbgso/prisma-generator-plantuml-erd.git"
48
+ },
49
+ "author": "osaki.shin <osaki.shin@dbgso.com>",
50
+ "keywords": [
51
+ "prisma",
52
+ "prisma2",
53
+ "generator",
54
+ "plantuml"
55
+ ],
56
+ "release": {
57
+ "branches": [
58
+ "main"
59
+ ],
60
+ "plugins": [
61
+ "@semantic-release/commit-analyzer",
62
+ "@semantic-release/release-notes-generator",
63
+ [
64
+ "@semantic-release/changelog",
65
+ {
66
+ "changelogFile": "CHANGELOG.md"
67
+ }
68
+ ],
69
+ "@semantic-release/npm",
70
+ "@semantic-release/github",
71
+ [
72
+ "@semantic-release/git",
73
+ {
74
+ "assets": [
75
+ "CHANGELOG.md",
76
+ "package.json"
77
+ ],
78
+ "message": "chore(release): set `package.json` to ${nextRelease.version} [skip ci]\n\n${nextRelease.notes}"
79
+ }
80
+ ]
81
+ ]
82
+ }
83
+ }