@koalarx/nest-cli 3.0.3 → 3.0.4

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.
Files changed (90) hide show
  1. package/LICENSE +21 -0
  2. package/commands/new-project/index.js +70 -0
  3. package/commands/new-project/new-project.js +69 -0
  4. package/index.js +39 -0
  5. package/package.json +9 -3
  6. package/templates/startup-project/.prettierrc.json +9 -0
  7. package/templates/startup-project/eslint.config.mts +85 -0
  8. package/templates/{root-files-folders → startup-project}/package.json +1 -1
  9. package/templates/startup-project/prisma/migrations/20250320165043_init/migration.sql +19 -0
  10. package/templates/startup-project/prisma/migrations/20250320171528_aplicando_map_de_tabelas_e_colunas/migration.sql +35 -0
  11. package/templates/startup-project/prisma/migrations/20250321190248_incluindo_status_ao_person/migration.sql +2 -0
  12. package/templates/startup-project/prisma/migrations/20250326185507_incluindo_tabela_de_endereco/migration.sql +19 -0
  13. package/templates/startup-project/prisma/migrations/20250326195627_aplicando_on_delete_cascade/migration.sql +5 -0
  14. package/templates/startup-project/prisma/migrations/20250326200821_removendo_ondelete_cascade/migration.sql +5 -0
  15. package/templates/startup-project/prisma/migrations/migration_lock.toml +3 -0
  16. package/templates/startup-project/prisma/schema.prisma +39 -0
  17. package/templates/startup-project/prisma.config.ts +8 -0
  18. package/templates/startup-project/src/application/mapping/mapping.profile.ts +10 -0
  19. package/templates/startup-project/src/application/mapping/person.mapping.ts +39 -0
  20. package/templates/startup-project/src/application/person/common/persist-person.request.ts +28 -0
  21. package/templates/startup-project/src/application/person/create/create-person.handler.spec.ts +22 -0
  22. package/templates/startup-project/src/application/person/create/create-person.handler.ts +39 -0
  23. package/templates/startup-project/src/application/person/create/create-person.request.ts +11 -0
  24. package/templates/startup-project/src/application/person/create/create-person.response.ts +3 -0
  25. package/templates/startup-project/src/application/person/create/create-person.validator.ts +19 -0
  26. package/templates/startup-project/src/application/person/create-person-job/create-person-job.ts +58 -0
  27. package/templates/startup-project/src/application/person/delete/delete-person.handler.spec.ts +24 -0
  28. package/templates/startup-project/src/application/person/delete/delete-person.handler.ts +33 -0
  29. package/templates/startup-project/src/application/person/delete-inative-job/delete-inactive-job.ts +49 -0
  30. package/templates/startup-project/src/application/person/events/inactive-person/inactive-person-event.ts +3 -0
  31. package/templates/startup-project/src/application/person/events/inactive-person/inactive-person-handler.ts +27 -0
  32. package/templates/startup-project/src/application/person/events/person-event.job.ts +11 -0
  33. package/templates/startup-project/src/application/person/read/read-person.handler.spec.ts +28 -0
  34. package/templates/startup-project/src/application/person/read/read-person.handler.ts +37 -0
  35. package/templates/startup-project/src/application/person/read/read-person.response.ts +44 -0
  36. package/templates/startup-project/src/application/person/read-many/read-many-person.handler.spec.ts +67 -0
  37. package/templates/startup-project/src/application/person/read-many/read-many-person.handler.ts +47 -0
  38. package/templates/startup-project/src/application/person/read-many/read-many-person.request.ts +21 -0
  39. package/templates/startup-project/src/application/person/read-many/read-many-person.response.ts +16 -0
  40. package/templates/startup-project/src/application/person/read-many/read-many.validator.ts +16 -0
  41. package/templates/startup-project/src/application/person/update/update-person.handler.spec.ts +47 -0
  42. package/templates/startup-project/src/application/person/update/update-person.handler.ts +57 -0
  43. package/templates/startup-project/src/application/person/update/update-person.request.ts +36 -0
  44. package/templates/startup-project/src/application/person/update/update-person.validator.ts +21 -0
  45. package/templates/startup-project/src/core/env.ts +6 -0
  46. package/templates/startup-project/src/domain/dtos/read-many-person.dto.ts +16 -0
  47. package/templates/startup-project/src/domain/entities/person/person-address.ts +12 -0
  48. package/templates/startup-project/src/domain/entities/person/person-phone.ts +12 -0
  49. package/templates/startup-project/src/domain/entities/person/person.ts +24 -0
  50. package/templates/startup-project/src/domain/repositories/iperson.repository.ts +10 -0
  51. package/templates/startup-project/src/host/app.module.ts +19 -0
  52. package/templates/startup-project/src/host/controllers/controller.module.ts +16 -0
  53. package/templates/startup-project/src/host/controllers/person/create-person.controller.ts +30 -0
  54. package/templates/startup-project/src/host/controllers/person/delete-person.controller.ts +22 -0
  55. package/templates/startup-project/src/host/controllers/person/person.controller.e2e-spec.ts +158 -0
  56. package/templates/startup-project/src/host/controllers/person/person.module.ts +37 -0
  57. package/templates/startup-project/src/host/controllers/person/read-many-person.controller.ts +29 -0
  58. package/templates/startup-project/src/host/controllers/person/read-person.controller.ts +26 -0
  59. package/templates/startup-project/src/host/controllers/person/router.config.ts +9 -0
  60. package/templates/startup-project/src/host/controllers/person/update-person.controller.ts +30 -0
  61. package/templates/startup-project/src/host/main.ts +38 -0
  62. package/templates/startup-project/src/infra/database/db-transaction-context.ts +24 -0
  63. package/templates/startup-project/src/infra/database/repositories/person.repository.ts +54 -0
  64. package/templates/startup-project/src/infra/database/repositories/repositories.module.ts +15 -0
  65. package/templates/startup-project/src/infra/infra.module.ts +8 -0
  66. package/templates/startup-project/src/test/create-e2e-test-app.ts +27 -0
  67. package/templates/startup-project/src/test/create-unit-test-app.ts +24 -0
  68. package/templates/startup-project/src/test/mockup/person/create-person-request.mockup.ts +9 -0
  69. package/templates/startup-project/src/test/repositories/person.repository.ts +36 -0
  70. package/templates/startup-project/src/test/setup-e2e.ts +12 -0
  71. package/templates/startup-project/tsconfig.app.json +8 -0
  72. package/tsconfig.tsbuildinfo +1 -0
  73. package/CHANGELOG.md +0 -17
  74. package/src/commands/new-project/index.ts +0 -99
  75. package/src/commands/new-project/new-project.ts +0 -91
  76. package/src/index.ts +0 -51
  77. package/tsconfig.json +0 -14
  78. /package/templates/{root-files-folders → startup-project}/.dockerignore +0 -0
  79. /package/templates/{root-files-folders → startup-project}/.gitattributes +0 -0
  80. /package/templates/{root-files-folders → startup-project}/.vscode/launch.json +0 -0
  81. /package/templates/{root-files-folders → startup-project}/.vscode/mcp.json +0 -0
  82. /package/templates/{root-files-folders → startup-project}/.vscode/settings.json +0 -0
  83. /package/templates/{root-files-folders → startup-project}/.vscode/tasks.json +0 -0
  84. /package/templates/{root-files-folders → startup-project}/Dockerfile +0 -0
  85. /package/templates/{root-files-folders → startup-project}/README.md +0 -0
  86. /package/templates/{root-files-folders → startup-project}/bunfig.toml +0 -0
  87. /package/templates/{root-files-folders → startup-project}/nest-cli.json +0 -0
  88. /package/templates/{root-files-folders → startup-project}/tsconfig.build.json +0 -0
  89. /package/templates/{root-files-folders → startup-project}/tsconfig.json +0 -0
  90. /package/templates/{root-files-folders → startup-project}/tsconfig.prisma.json +0 -0
package/LICENSE ADDED
@@ -0,0 +1,21 @@
1
+ MIT License
2
+
3
+ Copyright (c) 2023 Igor D. Rangel
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
@@ -0,0 +1,70 @@
1
+ import chalk from 'chalk';
2
+ import { execSync } from 'node:child_process';
3
+ import { cpSync, readFileSync, writeFileSync } from 'node:fs';
4
+ import { dirname, join } from 'node:path';
5
+ import { fileURLToPath } from 'node:url';
6
+ const __filename = fileURLToPath(import.meta.url);
7
+ const __dirname = dirname(__filename);
8
+ const TEMPLATE_BASE = join(__dirname, '../../templates/startup-project');
9
+ const TEMPLATES_DIR = join(__dirname, '../../templates');
10
+ export async function newProject(projectName) {
11
+ const targetDir = join(process.cwd(), projectName);
12
+ console.log(chalk.blue('🚀 Criando projeto Koala Nest...'));
13
+ console.log(chalk.gray(`📁 Destino: ${targetDir}\n`));
14
+ console.log(chalk.yellow('📋 Copiando estrutura base...'));
15
+ cpSync(TEMPLATE_BASE, targetDir, {
16
+ recursive: true,
17
+ filter: (src) => {
18
+ const isBlacklisted = src.includes('node_modules') ||
19
+ (src.includes('dist') && !src.includes('dist-cli')) ||
20
+ src.includes('.git');
21
+ return !isBlacklisted;
22
+ }
23
+ });
24
+ console.log(chalk.yellow('📦 Adicionando configurações extras...'));
25
+ const dockerfilePath = join(TEMPLATES_DIR, 'startup-project', 'Dockerfile');
26
+ try {
27
+ cpSync(dockerfilePath, join(targetDir, 'Dockerfile'));
28
+ }
29
+ catch {
30
+ console.log(chalk.red('⚠️ Dockerfile não encontrado nos templates, pulando...'));
31
+ }
32
+ console.log(chalk.yellow('⚙️ Configurando package.json...'));
33
+ const packageJsonPath = join(targetDir, 'package.json');
34
+ const packageJson = JSON.parse(readFileSync(packageJsonPath, 'utf-8'));
35
+ packageJson.name = projectName;
36
+ writeFileSync(packageJsonPath, JSON.stringify(packageJson, null, 2));
37
+ console.log(chalk.yellow('📄 Atualizando README...'));
38
+ const readmePath = join(targetDir, 'README.md');
39
+ let readme = readFileSync(readmePath, 'utf-8');
40
+ readme = readme.replace(/\[projectName\]/g, projectName);
41
+ writeFileSync(readmePath, readme);
42
+ console.log(chalk.yellow('🔐 Criando arquivo .env...'));
43
+ const envTemplate = readFileSync(join(TEMPLATES_DIR, 'env', 'config.txt'), 'utf-8');
44
+ const envContent = envTemplate.replace(/\[projectName\]/g, projectName.replace(/-/g, '_'));
45
+ writeFileSync(join(targetDir, '.env'), envContent);
46
+ console.log(chalk.yellow('🚫 Criando .gitignore...'));
47
+ const gitIgnoreContent = readFileSync(join(TEMPLATES_DIR, 'gitignore', 'config.txt'), 'utf-8');
48
+ writeFileSync(join(targetDir, '.gitignore'), gitIgnoreContent);
49
+ console.log(chalk.yellow('\n📥 Instalando dependências...'));
50
+ try {
51
+ execSync(`cd ${projectName} && bun install`, {
52
+ stdio: 'inherit',
53
+ });
54
+ console.log(chalk.yellow('🔨 Gerando Prisma Client...'));
55
+ execSync(`cd ${projectName} && bun run prisma:generate`, {
56
+ stdio: 'inherit',
57
+ });
58
+ }
59
+ catch {
60
+ console.log(chalk.red('⚠️ Erro ao instalar dependências. Execute manualmente:'));
61
+ console.log(chalk.gray(` cd ${projectName}`));
62
+ console.log(chalk.gray(` bun install`));
63
+ console.log(chalk.gray(` bun run prisma:generate`));
64
+ }
65
+ console.log(chalk.green('\n✅ Projeto criado com sucesso!'));
66
+ console.log(chalk.cyan('\n📚 Próximos passos:'));
67
+ console.log(chalk.gray(` cd ${projectName}`));
68
+ console.log(chalk.gray(` bun run start:dev`));
69
+ console.log(chalk.gray(`\n📖 Documentação: https://github.com/igordrangel/koala-nest\n`));
70
+ }
@@ -0,0 +1,69 @@
1
+ import chalk from 'chalk';
2
+ import { execSync } from 'node:child_process';
3
+ import { cpSync, readFileSync, writeFileSync } from 'node:fs';
4
+ import { dirname, join } from 'node:path';
5
+ import { fileURLToPath } from 'node:url';
6
+ const __filename = fileURLToPath(import.meta.url);
7
+ const __dirname = dirname(__filename);
8
+ const TEMPLATE_BASE = join(__dirname, '../../../example');
9
+ const TEMPLATES_DIR = join(__dirname, '../../templates');
10
+ export async function newProject(projectName) {
11
+ const targetDir = join(process.cwd(), projectName);
12
+ console.log(chalk.blue('🚀 Criando projeto Koala Nest...'));
13
+ console.log(chalk.gray(`📁 Destino: ${targetDir}\n`));
14
+ console.log(chalk.yellow('📋 Copiando estrutura base...'));
15
+ cpSync(TEMPLATE_BASE, targetDir, {
16
+ recursive: true,
17
+ filter: (src) => {
18
+ return !src.includes('node_modules') &&
19
+ !src.includes('dist') &&
20
+ !src.includes('.git');
21
+ }
22
+ });
23
+ console.log(chalk.yellow('📦 Adicionando configurações extras...'));
24
+ const dockerfilePath = join(TEMPLATES_DIR, 'startup-project', 'Dockerfile');
25
+ try {
26
+ cpSync(dockerfilePath, join(targetDir, 'Dockerfile'));
27
+ }
28
+ catch {
29
+ console.log(chalk.red('⚠️ Dockerfile não encontrado nos templates, pulando...'));
30
+ }
31
+ console.log(chalk.yellow('⚙️ Configurando package.json...'));
32
+ const packageJsonPath = join(targetDir, 'package.json');
33
+ const packageJson = JSON.parse(readFileSync(packageJsonPath, 'utf-8'));
34
+ packageJson.name = projectName;
35
+ writeFileSync(packageJsonPath, JSON.stringify(packageJson, null, 2));
36
+ console.log(chalk.yellow('📄 Atualizando README...'));
37
+ const readmePath = join(targetDir, 'README.md');
38
+ let readme = readFileSync(readmePath, 'utf-8');
39
+ readme = readme.replace(/\[projectName\]/g, projectName);
40
+ writeFileSync(readmePath, readme);
41
+ console.log(chalk.yellow('🔐 Criando arquivo .env...'));
42
+ const envTemplate = readFileSync(join(TEMPLATES_DIR, 'env', 'config.txt'), 'utf-8');
43
+ const envContent = envTemplate.replace(/\[projectName\]/g, projectName.replace(/-/g, '_'));
44
+ writeFileSync(join(targetDir, '.env'), envContent);
45
+ console.log(chalk.yellow('🚫 Criando .gitignore...'));
46
+ const gitIgnoreContent = readFileSync(join(TEMPLATES_DIR, 'gitignore', 'config.txt'), 'utf-8');
47
+ writeFileSync(join(targetDir, '.gitignore'), gitIgnoreContent);
48
+ console.log(chalk.yellow('\n📥 Instalando dependências...'));
49
+ try {
50
+ execSync(`cd ${projectName} && bun install`, {
51
+ stdio: 'inherit',
52
+ });
53
+ console.log(chalk.yellow('🔨 Gerando Prisma Client...'));
54
+ execSync(`cd ${projectName} && bun build:prisma`, {
55
+ stdio: 'inherit',
56
+ });
57
+ }
58
+ catch {
59
+ console.log(chalk.red('⚠️ Erro ao instalar dependências. Execute manualmente:'));
60
+ console.log(chalk.gray(` cd ${projectName}`));
61
+ console.log(chalk.gray(` bun install`));
62
+ console.log(chalk.gray(` bun build:prisma`));
63
+ }
64
+ console.log(chalk.green('\n✅ Projeto criado com sucesso!'));
65
+ console.log(chalk.cyan('\n📚 Próximos passos:'));
66
+ console.log(chalk.gray(` cd ${projectName}`));
67
+ console.log(chalk.gray(` bun run start:dev`));
68
+ console.log(chalk.gray(`\n📖 Documentação: https://github.com/igordrangel/koala-nest\n`));
69
+ }
package/index.js ADDED
@@ -0,0 +1,39 @@
1
+ #!/usr/bin/env node
2
+ import program from 'commander';
3
+ import inquirer from 'inquirer';
4
+ import { newProject } from './commands/new-project/index.js';
5
+ import chalk from 'chalk';
6
+ import { readFileSync } from 'fs';
7
+ import { join, dirname } from 'path';
8
+ import { fileURLToPath } from 'url';
9
+ const __filename = fileURLToPath(import.meta.url);
10
+ const __dirname = dirname(__filename);
11
+ const packageJson = JSON.parse(readFileSync(join(__dirname, '../package.json'), 'utf-8'));
12
+ const banner = `
13
+ _ __ _ _ _ _ ____ _ ___
14
+ | |/ /___ __ _| | __ _ | \\ | | ___ ___| |_ / ___| | |_ _|
15
+ | ' // _ \\ / _\` | |/ _\` | | \\| |/ _ \\/ __| __| | | | | | |
16
+ | . \\ (_) | (_| | | (_| | | |\\ | __/\\__ \\ |_ | |___| |___ | |
17
+ |_|\\_\\___/ \\__,_|_|\\__,_| |_| \\_|\\___||___/\\__| \\____|_____|___|
18
+ `;
19
+ console.log(chalk.cyan(banner));
20
+ program.version(packageJson.version);
21
+ program
22
+ .command('new [projectName]')
23
+ .description('Cria um novo projeto Nest com Koala Nest')
24
+ .action(async (projectName) => {
25
+ if (!projectName) {
26
+ projectName = await inquirer
27
+ .prompt([
28
+ {
29
+ type: 'input',
30
+ name: 'projectName',
31
+ message: 'Informe o nome do projeto',
32
+ validate: (value) => value ? true : 'Não é permitido um nome vazio',
33
+ },
34
+ ])
35
+ .then((answers) => answers.projectName);
36
+ }
37
+ await newProject(projectName);
38
+ });
39
+ program.parse(process.argv);
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@koalarx/nest-cli",
3
- "version": "3.0.3",
3
+ "version": "3.0.4",
4
4
  "description": "CLI para criação de projetos utilizando Koala Nest",
5
5
  "main": "index.js",
6
6
  "bin": {
@@ -24,5 +24,11 @@
24
24
  "bugs": {
25
25
  "url": "https://github.com/igordrangel/koala-nest/issues"
26
26
  },
27
- "homepage": "https://github.com/igordrangel/koala-nest/tree/main/apps/koala-nest-cli#readme"
28
- }
27
+ "homepage": "https://github.com/igordrangel/koala-nest/tree/main/apps/koala-nest-cli#readme",
28
+ "dependencies": {
29
+ "chalk": "^2.4.2",
30
+ "commander": "^2.19.0",
31
+ "inquirer": "^6.2.2",
32
+ "shelljs": "^0.8.3"
33
+ }
34
+ }
@@ -0,0 +1,9 @@
1
+ {
2
+ "printWidth": 80,
3
+ "tabWidth": 2,
4
+ "singleQuote": true,
5
+ "trailingComma": "all",
6
+ "arrowParens": "always",
7
+ "semi": false,
8
+ "endOfLine": "lf"
9
+ }
@@ -0,0 +1,85 @@
1
+ import js from '@eslint/js'
2
+ import vitest from '@vitest/eslint-plugin'
3
+ import globals from 'globals'
4
+ import tseslint from 'typescript-eslint'
5
+ import prettierConfig from 'eslint-config-prettier'
6
+ import prettierPlugin from 'eslint-plugin-prettier'
7
+
8
+ export default [
9
+ {
10
+ ignores: [
11
+ '**/*.json',
12
+ 'node_modules',
13
+ 'dist',
14
+ 'coverage',
15
+ 'prisma/generated',
16
+ ],
17
+ },
18
+ js.configs.recommended,
19
+ ...tseslint.configs.recommended,
20
+ prettierConfig,
21
+ {
22
+ files: ['src/**/*.ts', 'scripts/*.mjs'],
23
+ languageOptions: {
24
+ parser: tseslint.parser,
25
+ parserOptions: {
26
+ ecmaVersion: 'latest',
27
+ sourceType: 'module',
28
+ },
29
+ globals: globals.node,
30
+ },
31
+ plugins: {
32
+ '@typescript-eslint': tseslint.plugin,
33
+ prettier: prettierPlugin,
34
+ },
35
+ rules: {
36
+ 'prettier/prettier': [
37
+ 'warn',
38
+ {
39
+ printWidth: 80,
40
+ tabWidth: 2,
41
+ singleQuote: true,
42
+ trailingComma: 'all',
43
+ arrowParens: 'always',
44
+ semi: false,
45
+ endOfLine: 'lf',
46
+ },
47
+ ],
48
+ 'no-unused-vars': 'off',
49
+ 'no-useless-constructor': 'off',
50
+ 'no-new': 'off',
51
+ 'no-use-before-define': 'off',
52
+ '@typescript-eslint/no-unused-vars': 'warn',
53
+ '@typescript-eslint/no-explicit-any': 'off',
54
+ '@typescript-eslint/no-unsafe-function-type': 'off',
55
+ '@typescript-eslint/ban-ts-comment': 'off',
56
+ '@typescript-eslint/no-unsafe-assignment': 'off',
57
+ '@typescript-eslint/no-unsafe-member-access': 'off',
58
+ '@typescript-eslint/no-unsafe-call': 'off',
59
+ 'lines-between-class-members': [
60
+ 'warn',
61
+ 'always',
62
+ { exceptAfterSingleLine: true },
63
+ ],
64
+ },
65
+ },
66
+ {
67
+ files: [
68
+ 'src/**/*.e2e-spec.ts',
69
+ 'src/**/*.spec.ts',
70
+ 'src/test/setup-e2e.ts',
71
+ ],
72
+ languageOptions: {
73
+ globals: {
74
+ ...vitest.environments.env.globals,
75
+ },
76
+ },
77
+ plugins: {
78
+ vitest,
79
+ },
80
+ rules: {
81
+ ...vitest.configs.recommended.rules,
82
+ 'vitest/no-conditional-expect': 'off',
83
+ },
84
+ },
85
+ ]
@@ -25,7 +25,7 @@
25
25
  },
26
26
  "dependencies": {
27
27
  "@koalarx/utils": "^4.2.1",
28
- "@koalarx/nest": "KOALARX_NEST_VERSION",
28
+ "@koalarx/nest": "^undefined",
29
29
  "@nestjs/common": "^11.0.12",
30
30
  "@nestjs/config": "^4.0.1",
31
31
  "@nestjs/core": "^11.0.12",
@@ -0,0 +1,19 @@
1
+ -- CreateTable
2
+ CREATE TABLE "Person" (
3
+ "id" SERIAL NOT NULL,
4
+ "name" TEXT NOT NULL,
5
+
6
+ CONSTRAINT "Person_pkey" PRIMARY KEY ("id")
7
+ );
8
+
9
+ -- CreateTable
10
+ CREATE TABLE "PersonPhone" (
11
+ "id" SERIAL NOT NULL,
12
+ "phone" TEXT NOT NULL,
13
+ "personId" INTEGER NOT NULL,
14
+
15
+ CONSTRAINT "PersonPhone_pkey" PRIMARY KEY ("id")
16
+ );
17
+
18
+ -- AddForeignKey
19
+ ALTER TABLE "PersonPhone" ADD CONSTRAINT "PersonPhone_personId_fkey" FOREIGN KEY ("personId") REFERENCES "Person"("id") ON DELETE CASCADE ON UPDATE CASCADE;
@@ -0,0 +1,35 @@
1
+ /*
2
+ Warnings:
3
+
4
+ - You are about to drop the `Person` table. If the table is not empty, all the data it contains will be lost.
5
+ - You are about to drop the `PersonPhone` table. If the table is not empty, all the data it contains will be lost.
6
+
7
+ */
8
+ -- DropForeignKey
9
+ ALTER TABLE "PersonPhone" DROP CONSTRAINT "PersonPhone_personId_fkey";
10
+
11
+ -- DropTable
12
+ DROP TABLE "Person";
13
+
14
+ -- DropTable
15
+ DROP TABLE "PersonPhone";
16
+
17
+ -- CreateTable
18
+ CREATE TABLE "person" (
19
+ "id" SERIAL NOT NULL,
20
+ "name" TEXT NOT NULL,
21
+
22
+ CONSTRAINT "person_pkey" PRIMARY KEY ("id")
23
+ );
24
+
25
+ -- CreateTable
26
+ CREATE TABLE "person_phone" (
27
+ "id" SERIAL NOT NULL,
28
+ "phone" TEXT NOT NULL,
29
+ "person_id" INTEGER NOT NULL,
30
+
31
+ CONSTRAINT "person_phone_pkey" PRIMARY KEY ("id")
32
+ );
33
+
34
+ -- AddForeignKey
35
+ ALTER TABLE "person_phone" ADD CONSTRAINT "person_phone_person_id_fkey" FOREIGN KEY ("person_id") REFERENCES "person"("id") ON DELETE CASCADE ON UPDATE CASCADE;
@@ -0,0 +1,2 @@
1
+ -- AlterTable
2
+ ALTER TABLE "person" ADD COLUMN "active" BOOLEAN NOT NULL DEFAULT true;
@@ -0,0 +1,19 @@
1
+ /*
2
+ Warnings:
3
+
4
+ - Added the required column `person_address_id` to the `person` table without a default value. This is not possible if the table is not empty.
5
+
6
+ */
7
+ -- AlterTable
8
+ ALTER TABLE "person" ADD COLUMN "person_address_id" INTEGER NOT NULL;
9
+
10
+ -- CreateTable
11
+ CREATE TABLE "person_address" (
12
+ "id" SERIAL NOT NULL,
13
+ "address" TEXT NOT NULL,
14
+
15
+ CONSTRAINT "person_address_pkey" PRIMARY KEY ("id")
16
+ );
17
+
18
+ -- AddForeignKey
19
+ ALTER TABLE "person" ADD CONSTRAINT "person_person_address_id_fkey" FOREIGN KEY ("person_address_id") REFERENCES "person_address"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
@@ -0,0 +1,5 @@
1
+ -- DropForeignKey
2
+ ALTER TABLE "person" DROP CONSTRAINT "person_person_address_id_fkey";
3
+
4
+ -- AddForeignKey
5
+ ALTER TABLE "person" ADD CONSTRAINT "person_person_address_id_fkey" FOREIGN KEY ("person_address_id") REFERENCES "person_address"("id") ON DELETE CASCADE ON UPDATE CASCADE;
@@ -0,0 +1,5 @@
1
+ -- DropForeignKey
2
+ ALTER TABLE "person" DROP CONSTRAINT "person_person_address_id_fkey";
3
+
4
+ -- AddForeignKey
5
+ ALTER TABLE "person" ADD CONSTRAINT "person_person_address_id_fkey" FOREIGN KEY ("person_address_id") REFERENCES "person_address"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
@@ -0,0 +1,3 @@
1
+ # Please do not edit this file manually
2
+ # It should be added in your version-control system (e.g., Git)
3
+ provider = "postgresql"
@@ -0,0 +1,39 @@
1
+ generator client {
2
+ provider = "prisma-client"
3
+ output = "generated"
4
+ }
5
+
6
+ datasource db {
7
+ provider = "postgresql"
8
+ }
9
+
10
+ model Person {
11
+ id Int @id @default(autoincrement())
12
+ name String
13
+ active Boolean @default(true)
14
+ personAddressId Int @map("person_address_id")
15
+
16
+ phones PersonPhone[]
17
+ address PersonAddress @relation(fields: [personAddressId], references: [id])
18
+
19
+ @@map("person")
20
+ }
21
+
22
+ model PersonPhone {
23
+ id Int @id @default(autoincrement())
24
+ phone String
25
+
26
+ personId Int @map("person_id")
27
+ person Person @relation(fields: [personId], references: [id], onDelete: Cascade)
28
+
29
+ @@map("person_phone")
30
+ }
31
+
32
+ model PersonAddress {
33
+ id Int @id @default(autoincrement())
34
+ address String
35
+
36
+ people Person[]
37
+
38
+ @@map("person_address")
39
+ }
@@ -0,0 +1,8 @@
1
+ import 'dotenv/config'
2
+ import { defineConfig, env } from 'prisma/config'
3
+
4
+ export default defineConfig({
5
+ datasource: {
6
+ url: env('DATABASE_URL'),
7
+ },
8
+ })
@@ -0,0 +1,10 @@
1
+ import { AutoMappingProfile } from '@koalarx/nest/core/mapping/auto-mapping-profile'
2
+ import { Injectable } from '@nestjs/common'
3
+ import { PersonMapping } from './person.mapping'
4
+
5
+ @Injectable()
6
+ export class MappingProfile implements AutoMappingProfile {
7
+ profile(): void {
8
+ PersonMapping.createMap()
9
+ }
10
+ }
@@ -0,0 +1,39 @@
1
+ import { ReadManyPersonDto } from '@/domain/dtos/read-many-person.dto'
2
+ import { Person } from '@/domain/entities/person/person'
3
+ import { PersonAddress } from '@/domain/entities/person/person-address'
4
+ import { PersonPhone } from '@/domain/entities/person/person-phone'
5
+ import { createMap } from '@koalarx/nest/core/mapping/create-map'
6
+ import {
7
+ CreatePersonAddressRequest,
8
+ CreatePersonPhoneRequest,
9
+ CreatePersonRequest,
10
+ } from '../person/create/create-person.request'
11
+ import { ReadManyPersonRequest } from '../person/read-many/read-many-person.request'
12
+ import {
13
+ ReadPersonAddressResponse,
14
+ ReadPersonPhoneResponse,
15
+ ReadPersonResponse,
16
+ } from '../person/read/read-person.response'
17
+ import {
18
+ UpdatePersonAddressRequest,
19
+ UpdatePersonPhoneRequest,
20
+ UpdatePersonRequest,
21
+ } from '../person/update/update-person.request'
22
+
23
+ export class PersonMapping {
24
+ static createMap() {
25
+ createMap(CreatePersonAddressRequest, PersonAddress)
26
+ createMap(CreatePersonPhoneRequest, PersonPhone)
27
+ createMap(CreatePersonRequest, Person)
28
+
29
+ createMap(PersonAddress, ReadPersonAddressResponse)
30
+ createMap(PersonPhone, ReadPersonPhoneResponse)
31
+ createMap(Person, ReadPersonResponse)
32
+
33
+ createMap(ReadManyPersonRequest, ReadManyPersonDto)
34
+
35
+ createMap(UpdatePersonAddressRequest, PersonAddress)
36
+ createMap(UpdatePersonPhoneRequest, PersonPhone)
37
+ createMap(UpdatePersonRequest, Person)
38
+ }
39
+ }
@@ -0,0 +1,28 @@
1
+ import { AutoMap } from '@koalarx/nest/core/mapping/auto-mapping.decorator'
2
+ import { ApiProperty } from '@nestjs/swagger'
3
+
4
+ export class PersistPersonAddressRequest {
5
+ @ApiProperty({ example: 'Street 1' })
6
+ @AutoMap()
7
+ address: string
8
+ }
9
+
10
+ export class PersistPersonPhoneRequest {
11
+ @ApiProperty({ example: '22999999999' })
12
+ @AutoMap()
13
+ phone: string
14
+ }
15
+
16
+ export class PersistPersonRequest {
17
+ @ApiProperty({ example: 'John Doe' })
18
+ @AutoMap()
19
+ name: string
20
+
21
+ @ApiProperty({ type: [PersistPersonPhoneRequest] })
22
+ @AutoMap({ type: () => PersistPersonPhoneRequest, isArray: { addTo: true } })
23
+ phones: Array<PersistPersonPhoneRequest>
24
+
25
+ @ApiProperty({ type: PersistPersonAddressRequest })
26
+ @AutoMap({ type: () => PersistPersonAddressRequest })
27
+ address: PersistPersonAddressRequest
28
+ }
@@ -0,0 +1,22 @@
1
+ import { createUnitTestApp } from '@/test/create-unit-test-app'
2
+ import { createPersonRequestMockup } from '@/test/mockup/person/create-person-request.mockup'
3
+ import { CreatePersonHandler } from './create-person.handler'
4
+
5
+ describe('CreatePersonHandler', () => {
6
+ const app = createUnitTestApp()
7
+
8
+ it('should create a person', async () => {
9
+ const handler = app.get(CreatePersonHandler)
10
+ const request = createPersonRequestMockup
11
+
12
+ const result = await handler.handle(request)
13
+
14
+ expect(result.isOk()).toBeTruthy()
15
+
16
+ if (result.isOk()) {
17
+ expect(result.value).toEqual({
18
+ id: expect.any(Number),
19
+ })
20
+ }
21
+ })
22
+ })
@@ -0,0 +1,39 @@
1
+ import { Person } from '@/domain/entities/person/person'
2
+ import { IPersonRepository } from '@/domain/repositories/iperson.repository'
3
+ import { AutoMappingService } from '@koalarx/nest/core/mapping/auto-mapping.service'
4
+ import { RequestHandlerBase } from '@koalarx/nest/core/request-overflow/request-handler.base'
5
+ import {
6
+ ok,
7
+ RequestResult,
8
+ } from '@koalarx/nest/core/request-overflow/request-result'
9
+ import { Injectable } from '@nestjs/common'
10
+ import { CreatePersonRequest } from './create-person.request'
11
+ import { CreatePersonResponse } from './create-person.response'
12
+ import { CreatePersonValidator } from './create-person.validator'
13
+
14
+ @Injectable()
15
+ export class CreatePersonHandler extends RequestHandlerBase<
16
+ CreatePersonRequest,
17
+ RequestResult<Error, CreatePersonResponse>
18
+ > {
19
+ constructor(
20
+ private readonly mapper: AutoMappingService,
21
+ private readonly repository: IPersonRepository,
22
+ ) {
23
+ super()
24
+ }
25
+
26
+ async handle(
27
+ req: CreatePersonRequest,
28
+ ): Promise<RequestResult<Error, CreatePersonResponse>> {
29
+ const person = this.mapper.map(
30
+ new CreatePersonValidator(req).validate(),
31
+ CreatePersonRequest,
32
+ Person,
33
+ )
34
+
35
+ const result = await this.repository.save(person)
36
+
37
+ return ok({ id: result.id })
38
+ }
39
+ }
@@ -0,0 +1,11 @@
1
+ import {
2
+ PersistPersonAddressRequest,
3
+ PersistPersonPhoneRequest,
4
+ PersistPersonRequest,
5
+ } from '../common/persist-person.request'
6
+
7
+ export class CreatePersonAddressRequest extends PersistPersonAddressRequest {}
8
+
9
+ export class CreatePersonPhoneRequest extends PersistPersonPhoneRequest {}
10
+
11
+ export class CreatePersonRequest extends PersistPersonRequest {}
@@ -0,0 +1,3 @@
1
+ import { CreatedRegistreWithIdResponse } from '@koalarx/nest/core/controllers/created-registre-response.base'
2
+
3
+ export class CreatePersonResponse extends CreatedRegistreWithIdResponse {}
@@ -0,0 +1,19 @@
1
+ import { RequestValidatorBase } from '@koalarx/nest/core/request-overflow/request-validator.base'
2
+ import { z, ZodType, ZodTypeDef } from 'zod'
3
+ import { CreatePersonRequest } from './create-person.request'
4
+
5
+ export class CreatePersonValidator extends RequestValidatorBase<CreatePersonRequest> {
6
+ protected get schema(): ZodType<any, ZodTypeDef, any> {
7
+ return z.object({
8
+ name: z.string(),
9
+ phones: z.array(
10
+ z.object({
11
+ phone: z.string(),
12
+ }),
13
+ ),
14
+ address: z.object({
15
+ address: z.string(),
16
+ }),
17
+ })
18
+ }
19
+ }