From 7dc9126ab36b847e9cf695ce0c4b329dce791424 Mon Sep 17 00:00:00 2001 From: Milton Howe Date: Tue, 12 Sep 2017 15:19:55 -0400 Subject: [PATCH 1/2] add option to generate entities cased for Angular style guide (Pascal Entities, camel propertis, kebab files) --- README.md | 1 + package.json | 1 + src/Engine.ts | 24 ++++++++++++-- src/entity.mst | 48 +++++++++++++++------------- src/index.ts | 7 +++- src/models/EntityInfo.ts | 24 ++------------ test/integration/integration.test.ts | 12 ++++--- 7 files changed, 64 insertions(+), 53 deletions(-) diff --git a/README.md b/README.md index 2e85bf8..d8862a5 100644 --- a/README.md +++ b/README.md @@ -33,6 +33,7 @@ Options: -e, --engine Database engine. [choices: "mssql", "postgres", "mysql", "mariadb"] [default: "mssql"] -o, --output Where to place generated models. + -c, --case Convert snake_case tables names to PascalCase entities and snake_case columns to camelCase properties ``` ### Examples diff --git a/package.json b/package.json index 1c01f17..5755d9a 100644 --- a/package.json +++ b/package.json @@ -26,6 +26,7 @@ "dependencies": { "@types/mysql": "0.0.34", "@types/pg": "^6.1.41", + "change-case": "^3.0.1", "handlebars": "^4.0.10", "mssql": "^3.3.0", "mysql": "^2.14.1", diff --git a/src/Engine.ts b/src/Engine.ts index a613511..f473945 100644 --- a/src/Engine.ts +++ b/src/Engine.ts @@ -3,6 +3,7 @@ import { DatabaseModel } from './models/DatabaseModel' import * as Handlebars from 'handlebars' import fs = require('fs'); import path = require('path') +import changeCase = require("change-case"); /** * Engine */ @@ -24,6 +25,10 @@ export class Engine { } private createModelFromMetadata(databaseModel: DatabaseModel) { + Handlebars.registerHelper("curly", (open) => {return open ? "{" : "}";}); + Handlebars.registerHelper("toEntityName", str => {return this.Options.convertCase ? changeCase.pascalCase(str) : str;}); + Handlebars.registerHelper("toFileName", str => {return this.Options.convertCase ? changeCase.paramCase(str) : str;}); + Handlebars.registerHelper("toPropertyName", str => {return this.Options.convertCase ? changeCase.camelCase(str) : str;}); let templatePath = path.resolve(__dirname, '../../src/entity.mst') let template = fs.readFileSync(templatePath, 'UTF-8'); let resultPath = this.Options.resultsPath @@ -39,7 +44,19 @@ export class Engine { fs.mkdirSync(entitesPath); let compliedTemplate = Handlebars.compile(template,{noEscape:true}) databaseModel.entities.forEach(element => { - let resultFilePath = path.resolve(entitesPath, element.EntityName + '.ts'); + element.Imports = []; + element.Columns.forEach((column) => { + column.relations.forEach( + (relation) => { + if (element.EntityName !== relation.relatedTable) + {element.Imports.push(relation.relatedTable);} + } + ); + }); + element.Imports.filter(function (elem, index, self) { + return index === self.indexOf(elem); + }); + let resultFilePath = path.resolve(entitesPath, (this.Options.convertCase ? changeCase.paramCase(element.EntityName) : element.EntityName) + '.ts'); let rendered =compliedTemplate(element) fs.writeFileSync(resultFilePath, rendered, { encoding: 'UTF-8', flag: 'w' }) }); @@ -81,5 +98,6 @@ export interface EngineOptions { user: string, password: string, resultsPath: string, - databaseType: string -} \ No newline at end of file + databaseType: string, + convertCase: boolean +} diff --git a/src/entity.mst b/src/entity.mst index eef4c55..43ce01d 100644 --- a/src/entity.mst +++ b/src/entity.mst @@ -1,28 +1,30 @@ import {Index,Entity, PrimaryColumn, Column, OneToOne, OneToMany, ManyToOne, JoinColumn} from "typeorm"; -{{relationImports}} +{{#each Imports}} +import {{curly true}}{{toEntityName this}}{{curly false}} from "./{{toFileName this}}"; +{{/each}} -@Entity() +@Entity("{{toEntityName EntityName}}") {{#Indexes}}{{^isPrimaryKey}}@Index("{{name}}",[{{#columns}}"{{name}}",{{/columns}}]{{#isUnique}},{unique:true}{{/isUnique}}) -{{/isPrimaryKey}}{{/Indexes}}export class {{EntityName}} { +{{/isPrimaryKey}}{{/Indexes}}export class {{toEntityName EntityName}} { +{{#Columns}} - {{#Columns}} +{{~^relations}} @Column("{{sql_type}}", {{curly true}}{{#is_generated}}generated: true,{{/is_generated}} + name: "{{name}}",{{#is_nullable}} + nullable: true,{{/is_nullable}}{{^is_nullable}} + nullable: false,{{/is_nullable}}{{#char_max_lenght}} + length: {{.}},{{/char_max_lenght}}{{#default}} + default: "{{.}}",{{/default}}{{#numericPrecision}} + precision:{{.}},{{/numericPrecision}}{{#numericScale}} + scale: {{.}},{{/numericScale}}{{#isPrimary}} + primary: {{isPrimary}},{{/isPrimary}} + }) + {{toPropertyName name}}: {{ts_type}}; +{{/relations~}} - {{^relations}} @Column("{{sql_type}}",{ {{#is_generated}} - generated:true,{{/is_generated}}{{#is_nullable}} - nullable:true,{{/is_nullable}}{{^is_nullable}} - nullable:false,{{/is_nullable}}{{#char_max_lenght}} - length:{{.}},{{/char_max_lenght}}{{#default}} - default:"{{.}}",{{/default}}{{#numericPrecision}} - precision:{{.}},{{/numericPrecision}}{{#numericScale}} - scale:{{.}},{{/numericScale}}{{#isPrimary}} - primary:{{isPrimary}},{{/isPrimary}} - }) - {{name}}:{{ts_type}}; - {{/relations}}{{#relations}} - @{{relationType}}(type=>{{relatedTable}}, {{../name}}=>{{../name}}.{{#if isOwner}}{{ownerColumn}}{{else}}{{relatedColumn}}{{/if}}){{#isOwner}} - @JoinColumn(){{/isOwner}} - {{#if isOneToMany}}{{../name}}:{{relatedTable}}[]; - {{else}}{{../name}}:{{relatedTable}}; - {{/if}}{{/relations}} - {{/Columns}} -} \ No newline at end of file +{{~#relations}} @{{relationType}}(type => {{toEntityName relatedTable}}, {{toPropertyName ../name}} => {{toPropertyName ../name}}.{{#if isOwner}}{{toPropertyName ownerColumn}}{{else}}{{toPropertyName relatedColumn}}{{/if}}){{#isOwner}} + @JoinColumn(){{/isOwner}} + {{#if isOneToMany}}{{toPropertyName ../name}}: {{toEntityName relatedTable}}[];{{else}}{{toPropertyName ../name}}: {{toEntityName relatedTable}};{{/if}} +{{/relations~}} +{{"\n"}} +{{/Columns}} +} diff --git a/src/index.ts b/src/index.ts index 1e18b40..cc602e4 100644 --- a/src/index.ts +++ b/src/index.ts @@ -46,6 +46,10 @@ var argv = Yargs describe: 'Where to place generated models.', default: path.resolve(process.cwd(), 'output') }) + .option('c', { + alias: 'case', + describe: 'Convert snake_case tables names to PascalCase entities and snake_case columns to camelCase properties' + }) .argv; @@ -82,7 +86,8 @@ let engine = new Engine( user: argv.u, password: argv.x, databaseType: argv.e, - resultsPath: argv.o + resultsPath: argv.o, + convertCase: !!argv.c }); console.log(`[${new Date().toLocaleTimeString()}] Starting creation of model classes.`); diff --git a/src/models/EntityInfo.ts b/src/models/EntityInfo.ts index 68c4a3a..31d34d8 100644 --- a/src/models/EntityInfo.ts +++ b/src/models/EntityInfo.ts @@ -5,26 +5,6 @@ import { ColumnInfo } from './ColumnInfo' export class EntityInfo { EntityName: string; Columns: ColumnInfo[]; + Imports: string[]; Indexes: IndexInfo[]; - - relationImports(): any { - var returnString = ""; - var imports: string[] = []; - this.Columns.forEach((column) => { - column.relations.forEach( - (relation) => { - if (this.EntityName!=relation.relatedTable) - imports.push(relation.relatedTable); - } - ) - }); - imports.filter(function (elem, index, self) { - return index == self.indexOf(elem); - }).forEach((imp)=>{ - returnString+=`import {${imp}} from './${imp}'\n` - }) - - return returnString; - } - -} \ No newline at end of file +} diff --git a/test/integration/integration.test.ts b/test/integration/integration.test.ts index ce5cb09..9db9050 100644 --- a/test/integration/integration.test.ts +++ b/test/integration/integration.test.ts @@ -141,7 +141,8 @@ async function createMSSQLModels(filesOrgPath: string, resultsPath: string): Pro user: process.env.MSSQL_Username, password: process.env.MSSQL_Password, databaseType: 'mssql', - resultsPath: resultsPath + resultsPath: resultsPath, + convertCase: true }); @@ -184,7 +185,8 @@ async function createPostgresModels(filesOrgPath: string, resultsPath: string): user: process.env.POSTGRES_Username, password: process.env.POSTGRES_Password, databaseType: 'postgres', - resultsPath: resultsPath + resultsPath: resultsPath, + convertCase: true }); @@ -228,7 +230,8 @@ async function createMysqlModels(filesOrgPath: string, resultsPath: string): Pro user: process.env.MYSQL_Username, password: process.env.MYSQL_Password, databaseType: 'mysql', - resultsPath: resultsPath + resultsPath: resultsPath, + convertCase: true }); @@ -271,7 +274,8 @@ async function createMariaDBModels(filesOrgPath: string, resultsPath: string): P user: process.env.MARIADB_Username, password: process.env.MARIADB_Password, databaseType: 'mariadb', - resultsPath: resultsPath + resultsPath: resultsPath, + convertCase: true }); From f3458dbbe8851ee041682eefced4db55419e94bd Mon Sep 17 00:00:00 2001 From: Milton Howe Date: Mon, 15 Jan 2018 15:13:28 -0500 Subject: [PATCH 2/2] rebase to upstream master --- .idea/codeStyleSettings.xml | 9 + .idea/codeStyles/codeStyleConfig.xml | 5 + .idea/encodings.xml | 6 + .idea/inspectionProfiles/Project_Default.xml | 6 + .idea/jsLinters/tslint.xml | 4 + .idea/modules.xml | 8 + .idea/typeorm-model-generator.iml | 12 + .idea/vcs.xml | 6 + .idea/workspace.xml | 837 +++++++++ .travis.yml | 44 +- README.md | 11 +- docker-compose.yml | 31 +- package-lock.json | 1583 ++++++++++++----- package.json | 57 +- src/Engine.ts | 8 +- src/drivers/AbstractDriver.ts | 22 +- src/drivers/MariaDbDriver.ts | 401 +---- src/drivers/MssqlDriver.ts | 131 +- src/drivers/MysqlDriver.ts | 134 +- src/drivers/PostgresDriver.ts | 214 ++- src/entity.mst | 2 +- src/index.ts | 15 +- src/models/ColumnInfo.ts | 6 +- test/drivers/MssqlDriver.test.ts | 34 +- test/integration/entityTypes.test.ts | 99 ++ .../entityTypes/mariadb/entity/Post.ts | 94 + .../entityTypes/mssql/entity/Post.ts | 121 ++ .../entityTypes/mysql/entity/Post.ts | 93 + .../entityTypes/postgres/entity/Post.ts | 240 +++ .../sample1-simple-entity/entity/Post.ts | 4 +- .../entity/EverythingEntity.ts | 38 +- .../github-issues/12/entity/Post.ts | 16 + .../github-issues/12/entity/PostAuthor.ts | 16 + test/integration/githubIssues.test.ts | 100 ++ test/integration/integration.test.ts | 226 +-- test/utils/EntityFileToJson.ts | 14 + test/utils/GeneralTestUtils.ts | 206 +++ tsconfig.json | 3 +- typings.json | 3 +- 39 files changed, 3553 insertions(+), 1306 deletions(-) create mode 100644 .idea/codeStyleSettings.xml create mode 100644 .idea/codeStyles/codeStyleConfig.xml create mode 100644 .idea/encodings.xml create mode 100644 .idea/inspectionProfiles/Project_Default.xml create mode 100644 .idea/jsLinters/tslint.xml create mode 100644 .idea/modules.xml create mode 100644 .idea/typeorm-model-generator.iml create mode 100644 .idea/vcs.xml create mode 100644 .idea/workspace.xml create mode 100644 test/integration/entityTypes.test.ts create mode 100644 test/integration/entityTypes/mariadb/entity/Post.ts create mode 100644 test/integration/entityTypes/mssql/entity/Post.ts create mode 100644 test/integration/entityTypes/mysql/entity/Post.ts create mode 100644 test/integration/entityTypes/postgres/entity/Post.ts create mode 100644 test/integration/github-issues/12/entity/Post.ts create mode 100644 test/integration/github-issues/12/entity/PostAuthor.ts create mode 100644 test/integration/githubIssues.test.ts create mode 100644 test/utils/GeneralTestUtils.ts diff --git a/.idea/codeStyleSettings.xml b/.idea/codeStyleSettings.xml new file mode 100644 index 0000000..5555dd2 --- /dev/null +++ b/.idea/codeStyleSettings.xml @@ -0,0 +1,9 @@ + + + + + + \ No newline at end of file diff --git a/.idea/codeStyles/codeStyleConfig.xml b/.idea/codeStyles/codeStyleConfig.xml new file mode 100644 index 0000000..a55e7a1 --- /dev/null +++ b/.idea/codeStyles/codeStyleConfig.xml @@ -0,0 +1,5 @@ + + + + \ No newline at end of file diff --git a/.idea/encodings.xml b/.idea/encodings.xml new file mode 100644 index 0000000..f2f7930 --- /dev/null +++ b/.idea/encodings.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/.idea/inspectionProfiles/Project_Default.xml b/.idea/inspectionProfiles/Project_Default.xml new file mode 100644 index 0000000..1beadde --- /dev/null +++ b/.idea/inspectionProfiles/Project_Default.xml @@ -0,0 +1,6 @@ + + + + \ No newline at end of file diff --git a/.idea/jsLinters/tslint.xml b/.idea/jsLinters/tslint.xml new file mode 100644 index 0000000..5d68de2 --- /dev/null +++ b/.idea/jsLinters/tslint.xml @@ -0,0 +1,4 @@ + + + + \ No newline at end of file diff --git a/.idea/modules.xml b/.idea/modules.xml new file mode 100644 index 0000000..2070abf --- /dev/null +++ b/.idea/modules.xml @@ -0,0 +1,8 @@ + + + + + + + + \ No newline at end of file diff --git a/.idea/typeorm-model-generator.iml b/.idea/typeorm-model-generator.iml new file mode 100644 index 0000000..24643cc --- /dev/null +++ b/.idea/typeorm-model-generator.iml @@ -0,0 +1,12 @@ + + + + + + + + + + + + \ No newline at end of file diff --git a/.idea/vcs.xml b/.idea/vcs.xml new file mode 100644 index 0000000..94a25f7 --- /dev/null +++ b/.idea/vcs.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/.idea/workspace.xml b/.idea/workspace.xml new file mode 100644 index 0000000..05d6099 --- /dev/null +++ b/.idea/workspace.xml @@ -0,0 +1,837 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + char_max_lenght + .mst + EntityName + lower + relationimports + relatedTable + imports + relations + ownertable + Handlebars + toLowerCase + changeCase + changeCase.param + relationImports + char + argv + engineoptions + convertCase + camelcase + sample1-s + integration.test + "boolean" + "bool" + bool + + + + + + + + + + + + + true + DEFINITION_ORDER + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +