diff --git a/package-lock.json b/package-lock.json index 8d8b32c..a4a38e5 100644 --- a/package-lock.json +++ b/package-lock.json @@ -17,6 +17,7 @@ "@nestjs/platform-express": "^11.0.1", "@nestjs/swagger": "^11.2.0", "@nestjs/typeorm": "^11.0.0", + "@sentry/nestjs": "^10.10.0", "bcrypt": "^6.0.0", "class-transformer": "^0.5.1", "class-validator": "^0.14.2", @@ -2618,6 +2619,534 @@ "npm": ">=5.10.0" } }, + "node_modules/@opentelemetry/api": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/api/-/api-1.9.0.tgz", + "integrity": "sha512-3giAOQvZiH5F9bMlMiv8+GSPMeqg0dbaeo58/0SlA9sxSqZhnUtxzX9/2FzyhS9sWQf5S0GJE0AKBrFqjpeYcg==", + "license": "Apache-2.0", + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/@opentelemetry/api-logs": { + "version": "0.203.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/api-logs/-/api-logs-0.203.0.tgz", + "integrity": "sha512-9B9RU0H7Ya1Dx/Rkyc4stuBZSGVQF27WigitInx2QQoj6KUpEFYPKoWjdFTunJYxmXmh17HeBvbMa1EhGyPmqQ==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/api": "^1.3.0" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/@opentelemetry/context-async-hooks": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/context-async-hooks/-/context-async-hooks-2.1.0.tgz", + "integrity": "sha512-zOyetmZppnwTyPrt4S7jMfXiSX9yyfF0hxlA8B5oo2TtKl+/RGCy7fi4DrBfIf3lCPrkKsRBWZZD7RFojK7FDg==", + "license": "Apache-2.0", + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": ">=1.0.0 <1.10.0" + } + }, + "node_modules/@opentelemetry/core": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/core/-/core-2.1.0.tgz", + "integrity": "sha512-RMEtHsxJs/GiHHxYT58IY57UXAQTuUnZVco6ymDEqTNlJKTimM4qPUPVe8InNFyBjhHBEAx4k3Q8LtNayBsbUQ==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/semantic-conventions": "^1.29.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": ">=1.0.0 <1.10.0" + } + }, + "node_modules/@opentelemetry/instrumentation": { + "version": "0.203.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation/-/instrumentation-0.203.0.tgz", + "integrity": "sha512-ke1qyM+3AK2zPuBPb6Hk/GCsc5ewbLvPNkEuELx/JmANeEp6ZjnZ+wypPAJSucTw0wvCGrUaibDSdcrGFoWxKQ==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/api-logs": "0.203.0", + "import-in-the-middle": "^1.8.1", + "require-in-the-middle": "^7.1.1" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-amqplib": { + "version": "0.50.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-amqplib/-/instrumentation-amqplib-0.50.0.tgz", + "integrity": "sha512-kwNs/itehHG/qaQBcVrLNcvXVPW0I4FCOVtw3LHMLdYIqD7GJ6Yv2nX+a4YHjzbzIeRYj8iyMp0Bl7tlkidq5w==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/core": "^2.0.0", + "@opentelemetry/instrumentation": "^0.203.0", + "@opentelemetry/semantic-conventions": "^1.27.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-connect": { + "version": "0.47.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-connect/-/instrumentation-connect-0.47.0.tgz", + "integrity": "sha512-pjenvjR6+PMRb6/4X85L4OtkQCootgb/Jzh/l/Utu3SJHBid1F+gk9sTGU2FWuhhEfV6P7MZ7BmCdHXQjgJ42g==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/core": "^2.0.0", + "@opentelemetry/instrumentation": "^0.203.0", + "@opentelemetry/semantic-conventions": "^1.27.0", + "@types/connect": "3.4.38" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-dataloader": { + "version": "0.21.1", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-dataloader/-/instrumentation-dataloader-0.21.1.tgz", + "integrity": "sha512-hNAm/bwGawLM8VDjKR0ZUDJ/D/qKR3s6lA5NV+btNaPVm2acqhPcT47l2uCVi+70lng2mywfQncor9v8/ykuyw==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/instrumentation": "^0.203.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-express": { + "version": "0.52.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-express/-/instrumentation-express-0.52.0.tgz", + "integrity": "sha512-W7pizN0Wh1/cbNhhTf7C62NpyYw7VfCFTYg0DYieSTrtPBT1vmoSZei19wfKLnrMsz3sHayCg0HxCVL2c+cz5w==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/core": "^2.0.0", + "@opentelemetry/instrumentation": "^0.203.0", + "@opentelemetry/semantic-conventions": "^1.27.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-fs": { + "version": "0.23.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-fs/-/instrumentation-fs-0.23.0.tgz", + "integrity": "sha512-Puan+QopWHA/KNYvDfOZN6M/JtF6buXEyD934vrb8WhsX1/FuM7OtoMlQyIqAadnE8FqqDL4KDPiEfCQH6pQcQ==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/core": "^2.0.0", + "@opentelemetry/instrumentation": "^0.203.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-generic-pool": { + "version": "0.47.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-generic-pool/-/instrumentation-generic-pool-0.47.0.tgz", + "integrity": "sha512-UfHqf3zYK+CwDwEtTjaD12uUqGGTswZ7ofLBEdQ4sEJp9GHSSJMQ2hT3pgBxyKADzUdoxQAv/7NqvL42ZI+Qbw==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/instrumentation": "^0.203.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-graphql": { + "version": "0.51.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-graphql/-/instrumentation-graphql-0.51.0.tgz", + "integrity": "sha512-LchkOu9X5DrXAnPI1+Z06h/EH/zC7D6sA86hhPrk3evLlsJTz0grPrkL/yUJM9Ty0CL/y2HSvmWQCjbJEz/ADg==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/instrumentation": "^0.203.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-hapi": { + "version": "0.50.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-hapi/-/instrumentation-hapi-0.50.0.tgz", + "integrity": "sha512-5xGusXOFQXKacrZmDbpHQzqYD1gIkrMWuwvlrEPkYOsjUqGUjl1HbxCsn5Y9bUXOCgP1Lj6A4PcKt1UiJ2MujA==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/core": "^2.0.0", + "@opentelemetry/instrumentation": "^0.203.0", + "@opentelemetry/semantic-conventions": "^1.27.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-http": { + "version": "0.203.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-http/-/instrumentation-http-0.203.0.tgz", + "integrity": "sha512-y3uQAcCOAwnO6vEuNVocmpVzG3PER6/YZqbPbbffDdJ9te5NkHEkfSMNzlC3+v7KlE+WinPGc3N7MR30G1HY2g==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/core": "2.0.1", + "@opentelemetry/instrumentation": "0.203.0", + "@opentelemetry/semantic-conventions": "^1.29.0", + "forwarded-parse": "2.1.2" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-http/node_modules/@opentelemetry/core": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/@opentelemetry/core/-/core-2.0.1.tgz", + "integrity": "sha512-MaZk9SJIDgo1peKevlbhP6+IwIiNPNmswNL4AF0WaQJLbHXjr9SrZMgS12+iqr9ToV4ZVosCcc0f8Rg67LXjxw==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/semantic-conventions": "^1.29.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": ">=1.0.0 <1.10.0" + } + }, + "node_modules/@opentelemetry/instrumentation-ioredis": { + "version": "0.51.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-ioredis/-/instrumentation-ioredis-0.51.0.tgz", + "integrity": "sha512-9IUws0XWCb80NovS+17eONXsw1ZJbHwYYMXiwsfR9TSurkLV5UNbRSKb9URHO+K+pIJILy9wCxvyiOneMr91Ig==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/instrumentation": "^0.203.0", + "@opentelemetry/redis-common": "^0.38.0", + "@opentelemetry/semantic-conventions": "^1.27.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-kafkajs": { + "version": "0.13.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-kafkajs/-/instrumentation-kafkajs-0.13.0.tgz", + "integrity": "sha512-FPQyJsREOaGH64hcxlzTsIEQC4DYANgTwHjiB7z9lldmvua1LRMVn3/FfBlzXoqF179B0VGYviz6rn75E9wsDw==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/instrumentation": "^0.203.0", + "@opentelemetry/semantic-conventions": "^1.30.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-knex": { + "version": "0.48.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-knex/-/instrumentation-knex-0.48.0.tgz", + "integrity": "sha512-V5wuaBPv/lwGxuHjC6Na2JFRjtPgstw19jTFl1B1b6zvaX8zVDYUDaR5hL7glnQtUSCMktPttQsgK4dhXpddcA==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/instrumentation": "^0.203.0", + "@opentelemetry/semantic-conventions": "^1.33.1" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-koa": { + "version": "0.51.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-koa/-/instrumentation-koa-0.51.0.tgz", + "integrity": "sha512-XNLWeMTMG1/EkQBbgPYzCeBD0cwOrfnn8ao4hWgLv0fNCFQu1kCsJYygz2cvKuCs340RlnG4i321hX7R8gj3Rg==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/core": "^2.0.0", + "@opentelemetry/instrumentation": "^0.203.0", + "@opentelemetry/semantic-conventions": "^1.27.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-lru-memoizer": { + "version": "0.48.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-lru-memoizer/-/instrumentation-lru-memoizer-0.48.0.tgz", + "integrity": "sha512-KUW29wfMlTPX1wFz+NNrmE7IzN7NWZDrmFWHM/VJcmFEuQGnnBuTIdsP55CnBDxKgQ/qqYFp4udQFNtjeFosPw==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/instrumentation": "^0.203.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-mongodb": { + "version": "0.56.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-mongodb/-/instrumentation-mongodb-0.56.0.tgz", + "integrity": "sha512-YG5IXUUmxX3Md2buVMvxm9NWlKADrnavI36hbJsihqqvBGsWnIfguf0rUP5Srr0pfPqhQjUP+agLMsvu0GmUpA==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/instrumentation": "^0.203.0", + "@opentelemetry/semantic-conventions": "^1.27.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-mongoose": { + "version": "0.50.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-mongoose/-/instrumentation-mongoose-0.50.0.tgz", + "integrity": "sha512-Am8pk1Ct951r4qCiqkBcGmPIgGhoDiFcRtqPSLbJrUZqEPUsigjtMjoWDRLG1Ki1NHgOF7D0H7d+suWz1AAizw==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/core": "^2.0.0", + "@opentelemetry/instrumentation": "^0.203.0", + "@opentelemetry/semantic-conventions": "^1.27.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-mysql": { + "version": "0.49.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-mysql/-/instrumentation-mysql-0.49.0.tgz", + "integrity": "sha512-QU9IUNqNsrlfE3dJkZnFHqLjlndiU39ll/YAAEvWE40sGOCi9AtOF6rmEGzJ1IswoZ3oyePV7q2MP8SrhJfVAA==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/instrumentation": "^0.203.0", + "@opentelemetry/semantic-conventions": "^1.27.0", + "@types/mysql": "2.15.27" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-mysql2": { + "version": "0.50.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-mysql2/-/instrumentation-mysql2-0.50.0.tgz", + "integrity": "sha512-PoOMpmq73rOIE3nlTNLf3B1SyNYGsp7QXHYKmeTZZnJ2Ou7/fdURuOhWOI0e6QZ5gSem18IR1sJi6GOULBQJ9g==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/instrumentation": "^0.203.0", + "@opentelemetry/semantic-conventions": "^1.27.0", + "@opentelemetry/sql-common": "^0.41.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-nestjs-core": { + "version": "0.49.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-nestjs-core/-/instrumentation-nestjs-core-0.49.0.tgz", + "integrity": "sha512-1R/JFwdmZIk3T/cPOCkVvFQeKYzbbUvDxVH3ShXamUwBlGkdEu5QJitlRMyVNZaHkKZKWgYrBarGQsqcboYgaw==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/instrumentation": "^0.203.0", + "@opentelemetry/semantic-conventions": "^1.30.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-pg": { + "version": "0.55.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-pg/-/instrumentation-pg-0.55.0.tgz", + "integrity": "sha512-yfJ5bYE7CnkW/uNsnrwouG/FR7nmg09zdk2MSs7k0ZOMkDDAE3WBGpVFFApGgNu2U+gtzLgEzOQG4I/X+60hXw==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/core": "^2.0.0", + "@opentelemetry/instrumentation": "^0.203.0", + "@opentelemetry/semantic-conventions": "^1.27.0", + "@opentelemetry/sql-common": "^0.41.0", + "@types/pg": "8.15.4", + "@types/pg-pool": "2.0.6" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-redis": { + "version": "0.51.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-redis/-/instrumentation-redis-0.51.0.tgz", + "integrity": "sha512-uL/GtBA0u72YPPehwOvthAe+Wf8k3T+XQPBssJmTYl6fzuZjNq8zTfxVFhl9nRFjFVEe+CtiYNT0Q3AyqW1Z0A==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/instrumentation": "^0.203.0", + "@opentelemetry/redis-common": "^0.38.0", + "@opentelemetry/semantic-conventions": "^1.27.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-tedious": { + "version": "0.22.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-tedious/-/instrumentation-tedious-0.22.0.tgz", + "integrity": "sha512-XrrNSUCyEjH1ax9t+Uo6lv0S2FCCykcF7hSxBMxKf7Xn0bPRxD3KyFUZy25aQXzbbbUHhtdxj3r2h88SfEM3aA==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/instrumentation": "^0.203.0", + "@opentelemetry/semantic-conventions": "^1.27.0", + "@types/tedious": "^4.0.14" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, + "node_modules/@opentelemetry/instrumentation-undici": { + "version": "0.14.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation-undici/-/instrumentation-undici-0.14.0.tgz", + "integrity": "sha512-2HN+7ztxAReXuxzrtA3WboAKlfP5OsPA57KQn2AdYZbJ3zeRPcLXyW4uO/jpLE6PLm0QRtmeGCmfYpqRlwgSwg==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/core": "^2.0.0", + "@opentelemetry/instrumentation": "^0.203.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.7.0" + } + }, + "node_modules/@opentelemetry/redis-common": { + "version": "0.38.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/redis-common/-/redis-common-0.38.0.tgz", + "integrity": "sha512-4Wc0AWURII2cfXVVoZ6vDqK+s5n4K5IssdrlVrvGsx6OEOKdghKtJZqXAHWFiZv4nTDLH2/2fldjIHY8clMOjQ==", + "license": "Apache-2.0", + "engines": { + "node": "^18.19.0 || >=20.6.0" + } + }, + "node_modules/@opentelemetry/resources": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/resources/-/resources-2.1.0.tgz", + "integrity": "sha512-1CJjf3LCvoefUOgegxi8h6r4B/wLSzInyhGP2UmIBYNlo4Qk5CZ73e1eEyWmfXvFtm1ybkmfb2DqWvspsYLrWw==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/core": "2.1.0", + "@opentelemetry/semantic-conventions": "^1.29.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": ">=1.3.0 <1.10.0" + } + }, + "node_modules/@opentelemetry/sdk-trace-base": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/sdk-trace-base/-/sdk-trace-base-2.1.0.tgz", + "integrity": "sha512-uTX9FBlVQm4S2gVQO1sb5qyBLq/FPjbp+tmGoxu4tIgtYGmBYB44+KX/725RFDe30yBSaA9Ml9fqphe1hbUyLQ==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/core": "2.1.0", + "@opentelemetry/resources": "2.1.0", + "@opentelemetry/semantic-conventions": "^1.29.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": ">=1.3.0 <1.10.0" + } + }, + "node_modules/@opentelemetry/semantic-conventions": { + "version": "1.37.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/semantic-conventions/-/semantic-conventions-1.37.0.tgz", + "integrity": "sha512-JD6DerIKdJGmRp4jQyX5FlrQjA4tjOw1cvfsPAZXfOOEErMUHjPcPSICS+6WnM0nB0efSFARh0KAZss+bvExOA==", + "license": "Apache-2.0", + "engines": { + "node": ">=14" + } + }, + "node_modules/@opentelemetry/sql-common": { + "version": "0.41.0", + "resolved": "https://registry.npmjs.org/@opentelemetry/sql-common/-/sql-common-0.41.0.tgz", + "integrity": "sha512-pmzXctVbEERbqSfiAgdes9Y63xjoOyXcD7B6IXBkVb+vbM7M9U98mn33nGXxPf4dfYR0M+vhcKRZmbSJ7HfqFA==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/core": "^2.0.0" + }, + "engines": { + "node": "^18.19.0 || >=20.6.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.1.0" + } + }, "node_modules/@paralleldrive/cuid2": { "version": "2.2.2", "resolved": "https://registry.npmjs.org/@paralleldrive/cuid2/-/cuid2-2.2.2.tgz", @@ -2651,6 +3180,50 @@ "url": "https://opencollective.com/pkgr" } }, + "node_modules/@prisma/instrumentation": { + "version": "6.14.0", + "resolved": "https://registry.npmjs.org/@prisma/instrumentation/-/instrumentation-6.14.0.tgz", + "integrity": "sha512-Po/Hry5bAeunRDq0yAQueKookW3glpP+qjjvvyOfm6dI2KG5/Y6Bgg3ahyWd7B0u2E+Wf9xRk2rtdda7ySgK1A==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/instrumentation": "^0.52.0 || ^0.53.0 || ^0.54.0 || ^0.55.0 || ^0.56.0 || ^0.57.0" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.8" + } + }, + "node_modules/@prisma/instrumentation/node_modules/@opentelemetry/api-logs": { + "version": "0.57.2", + "resolved": "https://registry.npmjs.org/@opentelemetry/api-logs/-/api-logs-0.57.2.tgz", + "integrity": "sha512-uIX52NnTM0iBh84MShlpouI7UKqkZ7MrUszTmaypHBu4r7NofznSnQRfJ+uUeDtQDj6w8eFGg5KBLDAwAPz1+A==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/api": "^1.3.0" + }, + "engines": { + "node": ">=14" + } + }, + "node_modules/@prisma/instrumentation/node_modules/@opentelemetry/instrumentation": { + "version": "0.57.2", + "resolved": "https://registry.npmjs.org/@opentelemetry/instrumentation/-/instrumentation-0.57.2.tgz", + "integrity": "sha512-BdBGhQBh8IjZ2oIIX6F2/Q3LKm/FDDKi6ccYKcBTeilh6SNdNKveDOLk73BkSJjQLJk6qe4Yh+hHw1UPhCDdrg==", + "license": "Apache-2.0", + "dependencies": { + "@opentelemetry/api-logs": "0.57.2", + "@types/shimmer": "^1.2.0", + "import-in-the-middle": "^1.8.1", + "require-in-the-middle": "^7.1.1", + "semver": "^7.5.2", + "shimmer": "^1.2.1" + }, + "engines": { + "node": ">=14" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.3.0" + } + }, "node_modules/@scarf/scarf": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/@scarf/scarf/-/scarf-1.4.0.tgz", @@ -2658,6 +3231,149 @@ "hasInstallScript": true, "license": "Apache-2.0" }, + "node_modules/@sentry/core": { + "version": "10.10.0", + "resolved": "https://registry.npmjs.org/@sentry/core/-/core-10.10.0.tgz", + "integrity": "sha512-4O1O6my/vYE98ZgfEuLEwOOuHzqqzfBT6IdRo1yiQM7/AXcmSl0H/k4HJtXCiCTiHm+veEuTDBHp0GQZmpIbtA==", + "license": "MIT", + "engines": { + "node": ">=18" + } + }, + "node_modules/@sentry/nestjs": { + "version": "10.10.0", + "resolved": "https://registry.npmjs.org/@sentry/nestjs/-/nestjs-10.10.0.tgz", + "integrity": "sha512-qMO2XbEdUKsuB0DypGlcMzPepI28RkVnnfVS2rTTFtieeQErHl5hX/NHj2oeQQRSW7lpEXbEp2e7wESgRU7MZg==", + "license": "MIT", + "dependencies": { + "@opentelemetry/api": "^1.9.0", + "@opentelemetry/core": "^2.0.0", + "@opentelemetry/instrumentation": "^0.203.0", + "@opentelemetry/instrumentation-nestjs-core": "0.49.0", + "@opentelemetry/semantic-conventions": "^1.34.0", + "@sentry/core": "10.10.0", + "@sentry/node": "10.10.0" + }, + "engines": { + "node": ">=18" + }, + "peerDependencies": { + "@nestjs/common": "^8.0.0 || ^9.0.0 || ^10.0.0 || ^11.0.0", + "@nestjs/core": "^8.0.0 || ^9.0.0 || ^10.0.0 || ^11.0.0" + } + }, + "node_modules/@sentry/node": { + "version": "10.10.0", + "resolved": "https://registry.npmjs.org/@sentry/node/-/node-10.10.0.tgz", + "integrity": "sha512-GdI/ELIipKhdL8gdvnRLtz1ItPzAXRCZrvTwGMd5C+kDRALakQIR7pONC9nf5TKCG2UaslHEX+2XDImorhM7OA==", + "license": "MIT", + "dependencies": { + "@opentelemetry/api": "^1.9.0", + "@opentelemetry/context-async-hooks": "^2.0.0", + "@opentelemetry/core": "^2.0.0", + "@opentelemetry/instrumentation": "^0.203.0", + "@opentelemetry/instrumentation-amqplib": "0.50.0", + "@opentelemetry/instrumentation-connect": "0.47.0", + "@opentelemetry/instrumentation-dataloader": "0.21.1", + "@opentelemetry/instrumentation-express": "0.52.0", + "@opentelemetry/instrumentation-fs": "0.23.0", + "@opentelemetry/instrumentation-generic-pool": "0.47.0", + "@opentelemetry/instrumentation-graphql": "0.51.0", + "@opentelemetry/instrumentation-hapi": "0.50.0", + "@opentelemetry/instrumentation-http": "0.203.0", + "@opentelemetry/instrumentation-ioredis": "0.51.0", + "@opentelemetry/instrumentation-kafkajs": "0.13.0", + "@opentelemetry/instrumentation-knex": "0.48.0", + "@opentelemetry/instrumentation-koa": "0.51.0", + "@opentelemetry/instrumentation-lru-memoizer": "0.48.0", + "@opentelemetry/instrumentation-mongodb": "0.56.0", + "@opentelemetry/instrumentation-mongoose": "0.50.0", + "@opentelemetry/instrumentation-mysql": "0.49.0", + "@opentelemetry/instrumentation-mysql2": "0.50.0", + "@opentelemetry/instrumentation-pg": "0.55.0", + "@opentelemetry/instrumentation-redis": "0.51.0", + "@opentelemetry/instrumentation-tedious": "0.22.0", + "@opentelemetry/instrumentation-undici": "0.14.0", + "@opentelemetry/resources": "^2.0.0", + "@opentelemetry/sdk-trace-base": "^2.0.0", + "@opentelemetry/semantic-conventions": "^1.34.0", + "@prisma/instrumentation": "6.14.0", + "@sentry/core": "10.10.0", + "@sentry/node-core": "10.10.0", + "@sentry/opentelemetry": "10.10.0", + "import-in-the-middle": "^1.14.2", + "minimatch": "^9.0.0" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/@sentry/node-core": { + "version": "10.10.0", + "resolved": "https://registry.npmjs.org/@sentry/node-core/-/node-core-10.10.0.tgz", + "integrity": "sha512-7jHM1Is0Si737SVA0sHPg7lj7OmKoNM+f7+E3ySvtHIUeSINZBLM6jg1q57R1kIg8eavpHXudYljRMpuv/8bYA==", + "license": "MIT", + "dependencies": { + "@sentry/core": "10.10.0", + "@sentry/opentelemetry": "10.10.0", + "import-in-the-middle": "^1.14.2" + }, + "engines": { + "node": ">=18" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.9.0", + "@opentelemetry/context-async-hooks": "^1.30.1 || ^2.0.0", + "@opentelemetry/core": "^1.30.1 || ^2.0.0", + "@opentelemetry/instrumentation": ">=0.57.1 <1", + "@opentelemetry/resources": "^1.30.1 || ^2.0.0", + "@opentelemetry/sdk-trace-base": "^1.30.1 || ^2.0.0", + "@opentelemetry/semantic-conventions": "^1.34.0" + } + }, + "node_modules/@sentry/node/node_modules/brace-expansion": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.2.tgz", + "integrity": "sha512-Jt0vHyM+jmUBqojB7E1NIYadt0vI0Qxjxd2TErW94wDz+E2LAm5vKMXXwg6ZZBTHPuUlDgQHKXvjGBdfcF1ZDQ==", + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0" + } + }, + "node_modules/@sentry/node/node_modules/minimatch": { + "version": "9.0.5", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz", + "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==", + "license": "ISC", + "dependencies": { + "brace-expansion": "^2.0.1" + }, + "engines": { + "node": ">=16 || 14 >=14.17" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/@sentry/opentelemetry": { + "version": "10.10.0", + "resolved": "https://registry.npmjs.org/@sentry/opentelemetry/-/opentelemetry-10.10.0.tgz", + "integrity": "sha512-EQ5/1Ps4n1JosmaDiFCyb5iByjjKja2pnmeMiLzTDZ5Zikjs/3GKzmh+SgTRFLOm6yKgQps0GdiCH2gxdrbONg==", + "license": "MIT", + "dependencies": { + "@sentry/core": "10.10.0" + }, + "engines": { + "node": ">=18" + }, + "peerDependencies": { + "@opentelemetry/api": "^1.9.0", + "@opentelemetry/context-async-hooks": "^1.30.1 || ^2.0.0", + "@opentelemetry/core": "^1.30.1 || ^2.0.0", + "@opentelemetry/sdk-trace-base": "^1.30.1 || ^2.0.0", + "@opentelemetry/semantic-conventions": "^1.34.0" + } + }, "node_modules/@sinclair/typebox": { "version": "0.34.38", "resolved": "https://registry.npmjs.org/@sinclair/typebox/-/typebox-0.34.38.tgz", @@ -2830,7 +3546,6 @@ "version": "3.4.38", "resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.38.tgz", "integrity": "sha512-K6uROf1LD88uDQqJCktA4yzL1YYAK6NgfsI0v/mTgyPKWsX1CnJ0XPSDhViejru1GcRkLWb8RlzFYJRqGUbaug==", - "dev": true, "license": "MIT", "dependencies": { "@types/node": "*" @@ -2972,6 +3687,15 @@ "dev": true, "license": "MIT" }, + "node_modules/@types/mysql": { + "version": "2.15.27", + "resolved": "https://registry.npmjs.org/@types/mysql/-/mysql-2.15.27.tgz", + "integrity": "sha512-YfWiV16IY0OeBfBCk8+hXKmdTKrKlwKN1MNKAPBu5JYxLwBEZl7QzeEpGnlZb3VMGJrrGmB84gXiH+ofs/TezA==", + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, "node_modules/@types/node": { "version": "22.17.0", "resolved": "https://registry.npmjs.org/@types/node/-/node-22.17.0.tgz", @@ -3013,6 +3737,26 @@ "@types/passport": "*" } }, + "node_modules/@types/pg": { + "version": "8.15.4", + "resolved": "https://registry.npmjs.org/@types/pg/-/pg-8.15.4.tgz", + "integrity": "sha512-I6UNVBAoYbvuWkkU3oosC8yxqH21f4/Jc4DK71JLG3dT2mdlGe1z+ep/LQGXaKaOgcvUrsQoPRqfgtMcvZiJhg==", + "license": "MIT", + "dependencies": { + "@types/node": "*", + "pg-protocol": "*", + "pg-types": "^2.2.0" + } + }, + "node_modules/@types/pg-pool": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/@types/pg-pool/-/pg-pool-2.0.6.tgz", + "integrity": "sha512-TaAUE5rq2VQYxab5Ts7WZhKNmuN78Q6PiFonTDdpbx8a1H0M1vhy3rhiMjl+e2iHmogyMw7jZF4FrE6eJUy5HQ==", + "license": "MIT", + "dependencies": { + "@types/pg": "*" + } + }, "node_modules/@types/qs": { "version": "6.14.0", "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.14.0.tgz", @@ -3050,6 +3794,12 @@ "@types/send": "*" } }, + "node_modules/@types/shimmer": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/@types/shimmer/-/shimmer-1.2.0.tgz", + "integrity": "sha512-UE7oxhQLLd9gub6JKIAhDq06T0F6FnztwMNRvYgjeQSBeMc1ZG/tA47EwfduvkuQS8apbkM/lpLpWsaCeYsXVg==", + "license": "MIT" + }, "node_modules/@types/stack-utils": { "version": "2.0.3", "resolved": "https://registry.npmjs.org/@types/stack-utils/-/stack-utils-2.0.3.tgz", @@ -3081,6 +3831,15 @@ "@types/superagent": "^8.1.0" } }, + "node_modules/@types/tedious": { + "version": "4.0.14", + "resolved": "https://registry.npmjs.org/@types/tedious/-/tedious-4.0.14.tgz", + "integrity": "sha512-KHPsfX/FoVbUGbyYvk1q9MMQHLPeRZhRJZdO45Q4YjvFkv4hMNghCWTvy7rdKessBsmtz4euWCWAB6/tVpI1Iw==", + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, "node_modules/@types/validator": { "version": "13.15.2", "resolved": "https://registry.npmjs.org/@types/validator/-/validator-13.15.2.tgz", @@ -3830,7 +4589,6 @@ "version": "8.15.0", "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.15.0.tgz", "integrity": "sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg==", - "devOptional": true, "license": "MIT", "bin": { "acorn": "bin/acorn" @@ -3839,6 +4597,15 @@ "node": ">=0.4.0" } }, + "node_modules/acorn-import-attributes": { + "version": "1.9.5", + "resolved": "https://registry.npmjs.org/acorn-import-attributes/-/acorn-import-attributes-1.9.5.tgz", + "integrity": "sha512-n02Vykv5uA3eHGM/Z2dQrcD56kL8TyDb2p1+0P83PClMnC/nc+anbQRhIOWnSq4Ke/KvDPrY3C9hDtC/A3eHnQ==", + "license": "MIT", + "peerDependencies": { + "acorn": "^8" + } + }, "node_modules/acorn-import-phases": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/acorn-import-phases/-/acorn-import-phases-1.0.4.tgz", @@ -5977,6 +6744,12 @@ "node": ">= 0.6" } }, + "node_modules/forwarded-parse": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/forwarded-parse/-/forwarded-parse-2.1.2.tgz", + "integrity": "sha512-alTFZZQDKMporBH77856pXgzhEzaUVmLCDk+egLgIgHst3Tpndzz8MnKe+GzRJRfvVdn69HhpW7cmXzvtLvJAw==", + "license": "MIT" + }, "node_modules/fresh": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/fresh/-/fresh-2.0.0.tgz", @@ -6421,6 +7194,24 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/import-in-the-middle": { + "version": "1.14.2", + "resolved": "https://registry.npmjs.org/import-in-the-middle/-/import-in-the-middle-1.14.2.tgz", + "integrity": "sha512-5tCuY9BV8ujfOpwtAGgsTx9CGUapcFMEEyByLv1B+v2+6DhAcw+Zr0nhQT7uwaZ7DiourxFEscghOR8e1aPLQw==", + "license": "Apache-2.0", + "dependencies": { + "acorn": "^8.14.0", + "acorn-import-attributes": "^1.9.5", + "cjs-module-lexer": "^1.2.2", + "module-details-from-path": "^1.0.3" + } + }, + "node_modules/import-in-the-middle/node_modules/cjs-module-lexer": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/cjs-module-lexer/-/cjs-module-lexer-1.4.3.tgz", + "integrity": "sha512-9z8TZaGM1pfswYeXrUpzPrkx8UnWYdhJclsiYMm6x/w5+nN+8Tf/LnAgfLGQCm59qAOxU8WwHEq2vNwF6i4j+Q==", + "license": "MIT" + }, "node_modules/import-local": { "version": "3.2.0", "resolved": "https://registry.npmjs.org/import-local/-/import-local-3.2.0.tgz", @@ -6497,6 +7288,21 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/is-core-module": { + "version": "2.16.1", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.1.tgz", + "integrity": "sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==", + "license": "MIT", + "dependencies": { + "hasown": "^2.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/is-extglob": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", @@ -8060,6 +8866,12 @@ "mkdirp": "bin/cmd.js" } }, + "node_modules/module-details-from-path": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/module-details-from-path/-/module-details-from-path-1.0.4.tgz", + "integrity": "sha512-EGWKgxALGMgzvxYF1UyGTy0HXX/2vHLkw6+NvDKW2jypWbHpjQuj4UMcqQWXHERJhVGKikolT06G3bcKe4fi7w==", + "license": "MIT" + }, "node_modules/ms": { "version": "2.1.3", "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", @@ -8519,6 +9331,12 @@ "node": ">=8" } }, + "node_modules/path-parse": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", + "license": "MIT" + }, "node_modules/path-scurry": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-2.0.0.tgz", @@ -9058,6 +9876,40 @@ "node": ">=0.10.0" } }, + "node_modules/require-in-the-middle": { + "version": "7.5.2", + "resolved": "https://registry.npmjs.org/require-in-the-middle/-/require-in-the-middle-7.5.2.tgz", + "integrity": "sha512-gAZ+kLqBdHarXB64XpAe2VCjB7rIRv+mU8tfRWziHRJ5umKsIHN2tLLv6EtMw7WCdP19S0ERVMldNvxYCHnhSQ==", + "license": "MIT", + "dependencies": { + "debug": "^4.3.5", + "module-details-from-path": "^1.0.3", + "resolve": "^1.22.8" + }, + "engines": { + "node": ">=8.6.0" + } + }, + "node_modules/resolve": { + "version": "1.22.10", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.10.tgz", + "integrity": "sha512-NPRy+/ncIMeDlTAsuqwKIiferiawhefFJtkNSW0qZJEqMEb+qBt/77B/jGeeek+F0uOeN05CDa6HXbbIgtVX4w==", + "license": "MIT", + "dependencies": { + "is-core-module": "^2.16.0", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + }, + "bin": { + "resolve": "bin/resolve" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/resolve-cwd": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-3.0.0.tgz", @@ -9340,6 +10192,12 @@ "node": ">=8" } }, + "node_modules/shimmer": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/shimmer/-/shimmer-1.2.1.tgz", + "integrity": "sha512-sQTKC1Re/rM6XyFM6fIAGHRPVGvyXfgzIDvzoq608vM+jeyVD0Tu1E6Np0Kc2zAIFWIj963V2800iF/9LPieQw==", + "license": "BSD-2-Clause" + }, "node_modules/side-channel": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz", @@ -9788,6 +10646,18 @@ "node": ">=8" } }, + "node_modules/supports-preserve-symlinks-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", + "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/swagger-ui-dist": { "version": "5.21.0", "resolved": "https://registry.npmjs.org/swagger-ui-dist/-/swagger-ui-dist-5.21.0.tgz", diff --git a/src/app.module.ts b/src/app.module.ts index b4cc9a1..a425e1f 100644 --- a/src/app.module.ts +++ b/src/app.module.ts @@ -12,6 +12,7 @@ import { APP_FILTER } from '@nestjs/core'; import { ParentsModule } from './routes/parents/parents.module'; import { AuthModule } from './routes/auth/auth.module'; import { SentryGlobalFilter } from '@sentry/nestjs/setup'; +import { AllExceptionsFilter } from './common/filters/all_exceptions.filters'; @Module({ imports: [ @@ -54,6 +55,10 @@ import { SentryGlobalFilter } from '@sentry/nestjs/setup'; provide: APP_FILTER, useClass: SentryGlobalFilter }, + { + provide: APP_FILTER, + useClass: AllExceptionsFilter, + } ], }) diff --git a/src/main.ts b/src/main.ts index b12818f..f07f148 100644 --- a/src/main.ts +++ b/src/main.ts @@ -6,11 +6,20 @@ import { DocumentBuilder } from '@nestjs/swagger'; import { AuthGuard } from './common/guards/auth.guard'; import { JwtService } from '@nestjs/jwt'; import { RolesGuard } from './common/guards/roles.guard'; +import { ValidationPipe } from '@nestjs/common'; async function bootstrap() { const app = await NestFactory.create(AppModule, - {logger: ['error', 'warn', 'log', 'debug', 'verbose']}); - app.enableCors(); + { logger: ['error', 'warn', 'log', 'debug', 'verbose'] }); + app.enableCors(); + + app.useGlobalPipes( + new ValidationPipe({ + whitelist: true, + forbidNonWhitelisted: true, + transform: true, + }) + ); const configService = app.get(ConfigService); diff --git a/src/routes/assistantes_maternelles/assistantes_maternelles.controller.ts b/src/routes/assistantes_maternelles/assistantes_maternelles.controller.ts index d522297..b2bfa6b 100644 --- a/src/routes/assistantes_maternelles/assistantes_maternelles.controller.ts +++ b/src/routes/assistantes_maternelles/assistantes_maternelles.controller.ts @@ -1,7 +1,14 @@ -import { Controller, Get, Post, Body, Patch, Param, Delete } from '@nestjs/common'; +import { + Controller, + Get, + Post, + Body, + Patch, + Param, + Delete, +} from '@nestjs/common'; import { AssistantesMaternellesService } from './assistantes_maternelles.service'; import { ApiBody, ApiResponse, ApiTags } from '@nestjs/swagger'; -import { BaseController } from 'src/common/base.controller'; import { AssistanteMaternelle } from 'src/entities/assistantes_maternelles.entity'; import { Roles } from 'src/common/decorators/roles.decorator'; import { RoleType } from 'src/entities/users.entity'; @@ -10,51 +17,48 @@ import { UpdateAssistanteDto } from '../user/dto/update_assistante.dto'; @ApiTags("Assistantes Maternelles") @Controller('assistantes-maternelles') -export class AssistantesMaternellesController extends BaseController { - constructor(private readonly assistantesMaternellesService: AssistantesMaternellesService) { - super(assistantesMaternellesService); - } +export class AssistantesMaternellesController { + constructor(private readonly assistantesMaternellesService: AssistantesMaternellesService) {} - @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) //Seul les utilisateurs super admin et gestionnaire peuvent accéder à cette route + @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) @ApiResponse({ status: 201, description: 'Assistante maternelle créée avec succès' }) - @ApiResponse({ status: 403, description: 'Accès refusé : Reservé aux super_admins et gestionnaires' }) + @ApiResponse({ status: 403, description: 'Accès refusé : Réservé aux super_admins et gestionnaires' }) @ApiBody({ type: CreateAssistanteDto }) @Post() create(@Body() dto: CreateAssistanteDto): Promise { return this.assistantesMaternellesService.create(dto); } - //Lister toutes les nounous - @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) //Seul les utilisateurs super admin et gestionnaire peuvent accéder à cette route + @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) @Get() @ApiResponse({ status: 200, description: 'Liste des assistantes maternelles' }) - @ApiResponse({ status: 403, description: 'Accès refusé : Reservé aux super_admins et gestionnaires' }) - override getAll(): Promise { + @ApiResponse({ status: 403, description: 'Accès refusé : Réservé aux super_admins et gestionnaires' }) + getAll(): Promise { return this.assistantesMaternellesService.findAll(); } - @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) //Seul les utilisateurs super admin et gestionnaire peuvent accéder à cette route + @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) @Get(':id') @ApiResponse({ status: 200, description: 'Détails de l\'assistante maternelle' }) @ApiResponse({ status: 404, description: 'Assistante maternelle non trouvée' }) - @ApiResponse({ status: 403, description: 'Accès refusé : Reservé aux super_admins et gestionnaires' }) + @ApiResponse({ status: 403, description: 'Accès refusé : Réservé aux super_admins et gestionnaires' }) getOne(@Param('id') user_id: string): Promise { return this.assistantesMaternellesService.findOne(user_id); } - @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) //Seul les utilisateurs super admin et gestionnaire peuvent accéder à cette route + @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) @ApiBody({ type: UpdateAssistanteDto }) @ApiResponse({ status: 200, description: 'Assistante maternelle mise à jour avec succès' }) - @ApiResponse({ status: 403, description: 'Accès refusé : Reservé aux super_admins et gestionnaires' }) + @ApiResponse({ status: 403, description: 'Accès refusé : Réservé aux super_admins et gestionnaires' }) @ApiResponse({ status: 404, description: 'Assistante maternelle non trouvée' }) @Patch(':id') update(@Param('id') id: string, @Body() dto: UpdateAssistanteDto): Promise { return this.assistantesMaternellesService.update(id, dto); } - @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) //Seul les utilisateurs super admin et gestionnaire peuvent accéder à cette route + @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) @ApiResponse({ status: 200, description: 'Assistante maternelle supprimée avec succès' }) - @ApiResponse({ status: 403, description: 'Accès refusé : Reservé aux super_admins et gestionnaires' }) + @ApiResponse({ status: 403, description: 'Accès refusé : Réservé aux super_admins et gestionnaires' }) @ApiResponse({ status: 404, description: 'Assistante maternelle non trouvée' }) @Delete(':id') remove(@Param('id') id: string): Promise { diff --git a/src/routes/assistantes_maternelles/assistantes_maternelles.service.ts b/src/routes/assistantes_maternelles/assistantes_maternelles.service.ts index e19fd78..91ea7cc 100644 --- a/src/routes/assistantes_maternelles/assistantes_maternelles.service.ts +++ b/src/routes/assistantes_maternelles/assistantes_maternelles.service.ts @@ -1,67 +1,77 @@ -import { BadRequestException, ConflictException, Injectable, NotFoundException } from '@nestjs/common'; -import { BaseService } from 'src/common/base.service'; +import { + BadRequestException, + ConflictException, + Injectable, + NotFoundException, +} from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; -import { DeepPartial, Repository } from 'typeorm'; +import { Repository } from 'typeorm'; import { RoleType, Users } from 'src/entities/users.entity'; -import { UpdateAssistanteDto } from '../user/dto/update_assistante.dto'; import { AssistanteMaternelle } from 'src/entities/assistantes_maternelles.entity'; import { CreateAssistanteDto } from '../user/dto/create_assistante.dto'; +import { UpdateAssistanteDto } from '../user/dto/update_assistante.dto'; @Injectable() -export class AssistantesMaternellesService extends BaseService { +export class AssistantesMaternellesService { constructor( @InjectRepository(AssistanteMaternelle) private readonly assistantesMaternelleRepository: Repository, + @InjectRepository(Users) private readonly usersRepository: Repository - ) { - super(assistantesMaternelleRepository); - } + ) {} - override async create(data: DeepPartial): Promise { - const dto = data as CreateAssistanteDto; + // Création d’une assistante maternelle + async create(dto: CreateAssistanteDto): Promise { const user = await this.usersRepository.findOneBy({ id: dto.user_id }); if (!user) throw new NotFoundException('Utilisateur introuvable'); - if (user.role !== RoleType.ASSISTANTE_MATERNELLE) throw new BadRequestException('Acces reserve aux assistantes maternelles'); + if (user.role !== RoleType.ASSISTANTE_MATERNELLE) { + throw new BadRequestException('Accès réservé aux assistantes maternelles'); + } + const exist = await this.assistantesMaternelleRepository.findOneBy({ user_id: dto.user_id }); - if (exist) throw new ConflictException('Assistante maternelle deja existante'); + if (exist) throw new ConflictException('Assistante maternelle déjà existante'); const entity = this.assistantesMaternelleRepository.create({ user_id: dto.user_id, user: { ...user, role: RoleType.ASSISTANTE_MATERNELLE }, approval_number: dto.numero_agrement, - birthdate: new Date(dto.date_naissance), + birthdate: dto.date_naissance ? new Date(dto.date_naissance) : undefined, birthplace_city: dto.ville_naissance, birthplace_country: dto.pays_naissance, nir: dto.nir_chiffre, max_children: dto.nb_max_enfants, biography: dto.biographie, available: dto.disponible ?? true, - city: dto.ville_residence, - + residence_city: dto.ville_residence, }); + return this.assistantesMaternelleRepository.save(entity); } - override async findAll(): Promise { + // Liste des assistantes maternelles + async findAll(): Promise { return this.assistantesMaternelleRepository.find({ relations: ['user'], }); } - override async findOne(user_id: string): Promise { - const assistante_maternelle = await this.assistantesMaternelleRepository.findOne({ + // Récupérer une assistante maternelle par user_id + async findOne(user_id: string): Promise { + const assistante = await this.assistantesMaternelleRepository.findOne({ where: { user_id }, relations: ['user'], }); - if (!assistante_maternelle) throw new NotFoundException('Assistante maternelle introuvable'); - return assistante_maternelle; + if (!assistante) throw new NotFoundException('Assistante maternelle introuvable'); + return assistante; } + // Mise à jour async update(id: string, dto: UpdateAssistanteDto): Promise { await this.assistantesMaternelleRepository.update(id, dto); return this.findOne(id); } + // Suppression async remove(id: string): Promise { await this.assistantesMaternelleRepository.delete(id); } diff --git a/src/routes/gestionnaires/gestionnaires.controller.ts b/src/routes/gestionnaires/gestionnaires.controller.ts index 4d15cfc..e244c57 100644 --- a/src/routes/gestionnaires/gestionnaires.controller.ts +++ b/src/routes/gestionnaires/gestionnaires.controller.ts @@ -1,55 +1,52 @@ -import { Controller, Get, Post, Body, Patch, Param, Delete, ForbiddenException } from '@nestjs/common'; +import { + Controller, + Get, + Post, + Body, + Patch, + Param, + Delete, +} from '@nestjs/common'; import { GestionnairesService } from './gestionnaires.service'; import { RoleType, Users } from 'src/entities/users.entity'; -import { ApiBody } from '@nestjs/swagger'; -import { BaseController } from 'src/common/base.controller'; import { Roles } from 'src/common/decorators/roles.decorator'; import { UpdateGestionnaireDto } from '../user/dto/update_gestionnaire.dto'; import { CreateGestionnaireDto } from '../user/dto/create_gestionnaire.dto'; @Controller('gestionnaires') -export class GestionnairesController extends BaseController { - constructor(private readonly gestionnairesService: GestionnairesService) { - super(gestionnairesService); - } +export class GestionnairesController { + constructor(private readonly gestionnairesService: GestionnairesService) { } @Roles(RoleType.SUPER_ADMIN) @Post() - create(@Body() createGestionnaireDto: CreateGestionnaireDto) { - return this.gestionnairesService.create(createGestionnaireDto); + create(@Body() dto: CreateGestionnaireDto): Promise { + return this.gestionnairesService.create(dto); } - //Lister tous les gestionnaires @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) @Get() getAll(): Promise { - //Dans gestionnaire service le findAll cherche deja les gestionnaires - const gestionnaire = this.gestionnairesService.findAll(); - return gestionnaire; + return this.gestionnairesService.findAll(); } - //Récupérer un gestionnaire @Roles(RoleType.GESTIONNAIRE, RoleType.SUPER_ADMIN) @Get(':id') - findOne(@Param('id') id: string) { + findOne(@Param('id') id: string): Promise { return this.gestionnairesService.findOne(id); } - - //Modifier un gestionnaire @Roles(RoleType.SUPER_ADMIN) @Patch(':id') update( @Param('id') id: string, @Body() dto: UpdateGestionnaireDto, - ) { - const update_gestionnaire = this.gestionnairesService.update(id, dto); - return update_gestionnaire; + ): Promise { + return this.gestionnairesService.update(id, dto); } @Roles(RoleType.SUPER_ADMIN) @Delete(':id') - remove(@Param('id') id: string) { + remove(@Param('id') id: string): Promise { return this.gestionnairesService.remove(id); } } diff --git a/src/routes/gestionnaires/gestionnaires.service.ts b/src/routes/gestionnaires/gestionnaires.service.ts index 601f6fd..43929d4 100644 --- a/src/routes/gestionnaires/gestionnaires.service.ts +++ b/src/routes/gestionnaires/gestionnaires.service.ts @@ -1,50 +1,88 @@ -import { ConflictException, ForbiddenException, Injectable, NotFoundException } from '@nestjs/common'; -import { BaseService } from 'src/common/base.service'; +import { + ConflictException, + ForbiddenException, + Injectable, + NotFoundException, +} from '@nestjs/common'; +import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; import { RoleType, Users } from 'src/entities/users.entity'; -import { InjectRepository } from '@nestjs/typeorm'; -import { UpdateGestionnaireDto } from '../user/dto/update_gestionnaire.dto'; import { CreateGestionnaireDto } from '../user/dto/create_gestionnaire.dto'; +import { UpdateGestionnaireDto } from '../user/dto/update_gestionnaire.dto'; +import * as bcrypt from 'bcrypt'; @Injectable() -export class GestionnairesService extends BaseService { +export class GestionnairesService { constructor( @InjectRepository(Users) private readonly gestionnaireRepository: Repository, - ) { - super(gestionnaireRepository); - } - override async create(dto: CreateGestionnaireDto): Promise { + ) { } + // Création d’un gestionnaire + async create(dto: CreateGestionnaireDto): Promise { const exist = await this.gestionnaireRepository.findOneBy({ email: dto.email }); - if (exist) throw new ConflictException('Email déjà utilise'); + if (exist) throw new ConflictException('Email déjà utilisé'); + + const salt = await bcrypt.genSalt(); + const password_hash = await bcrypt.hash(dto.password, salt); const entity = this.gestionnaireRepository.create({ - ...dto, + email: dto.email, + password_hash, + first_name: dto.first_name, + last_name: dto.last_name, + gender: dto.gender, + status: dto.status, + phone: dto.phone, + address: dto.address, + photo_url: dto.photo_url, + consent_photo: dto.consent_photo ?? false, + consent_photo_at: dto.consent_photo_at ? new Date(dto.consent_photo_at) : undefined, + must_change_password: dto.must_change_password ?? false, role: RoleType.GESTIONNAIRE, }); + return this.gestionnaireRepository.save(entity); } - override async findAll(): Promise { - // Retourner seulement les utilisateurs avec le rôle de gestionnaire - const all_gestionnaires = await this.gestionnaireRepository.find({ where: { role: RoleType.GESTIONNAIRE } }); - return all_gestionnaires; + // Liste des gestionnaires + async findAll(): Promise { + return this.gestionnaireRepository.find({ where: { role: RoleType.GESTIONNAIRE } }); } - override async findOne(id: string): Promise { - const gestionnaire = await this.gestionnaireRepository.findOne({ where: { id, role: RoleType.GESTIONNAIRE } }); + // Récupérer un gestionnaire par ID + async findOne(id: string): Promise { + const gestionnaire = await this.gestionnaireRepository.findOne({ + where: { id, role: RoleType.GESTIONNAIRE }, + }); if (!gestionnaire) throw new NotFoundException('Gestionnaire introuvable'); return gestionnaire; } - async update(id: string, dto: UpdateGestionnaireDto): Promise { - await this.gestionnaireRepository.update(id, dto); - return this.findOne(id); + // Mise à jour d’un gestionnaire + async update(id: string, dto: UpdateGestionnaireDto): Promise { + const gestionnaire = await this.findOne(id); + if (!gestionnaire) throw new NotFoundException('Gestionnaire introuvable'); + + if (dto.password) { + const salt = await bcrypt.genSalt(); + gestionnaire.password_hash = await bcrypt.hash(dto.password, salt); + } + + if (dto.consent_photo_at !== undefined) { + gestionnaire.consent_photo_at = dto.consent_photo_at + ? new Date(dto.consent_photo_at) + : undefined; + } + + Object.assign(gestionnaire, dto); + return this.gestionnaireRepository.save(gestionnaire); } + // Suppression d’un gestionnaire async remove(id: string): Promise { - await this.findOne(id); + const gestionnaire = await this.findOne(id); + if (!gestionnaire) throw new NotFoundException('Gestionnaire introuvable'); await this.gestionnaireRepository.delete(id); } } diff --git a/src/routes/parents/parents.controller.ts b/src/routes/parents/parents.controller.ts index d3f5a2b..9f1b607 100644 --- a/src/routes/parents/parents.controller.ts +++ b/src/routes/parents/parents.controller.ts @@ -1,73 +1,64 @@ -import { Body, Controller, Delete, Get, Param, Patch, Post } from '@nestjs/common'; -import { BaseController } from 'src/common/base.controller'; -import { Parents } from 'src/entities/parents.entity'; +import { + Body, + Controller, + Delete, + Get, + Param, + Patch, + Post, +} from '@nestjs/common'; import { ParentsService } from './parents.service'; +import { Parents } from 'src/entities/parents.entity'; import { Roles } from 'src/common/decorators/roles.decorator'; import { RoleType } from 'src/entities/users.entity'; -import * as typeorm from 'typeorm'; import { ApiBody, ApiResponse, ApiTags } from '@nestjs/swagger'; import { CreateParentDto } from '../user/dto/create_parent.dto'; import { UpdateParentsDto } from '../user/dto/update_parent.dto'; - @ApiTags('Parents') @Controller('parents') -export class ParentsController extends BaseController { - constructor(private readonly parentsService: ParentsService) { - super(parentsService); - } +export class ParentsController { + constructor(private readonly parentsService: ParentsService) {} - //Lister tous les parents - @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) //Seul les utilisateurs super admin et gestionnaire peuvent accéder à cette route - @Get() - @ApiResponse({ status: 200, description: 'Liste des parents' }) - @ApiResponse({ status: 403, description: 'Accès refusé : Reservé aux super_admins' }) - override getAll(): Promise { - return this.parentsService.findAll(); - } + @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) + @Get() + @ApiResponse({ status: 200, description: 'Liste des parents' }) + @ApiResponse({ status: 403, description: 'Accès refusé : Réservé aux super_admins' }) + getAll(): Promise { + return this.parentsService.findAll(); + } - //Rechercher par user_id - @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) //Seul les utilisateurs super admin et gestionnaire peuvent accéder à cette route - @Get(':id') - @ApiResponse({ status: 200, description: 'Détails du parent par ID utilisateur' }) - @ApiResponse({ status: 404, description: 'Parent non trouvé' }) - getOne(@Param('id') user_id: string): Promise { - return this.parentsService.findOne(user_id); - } + @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) + @Get(':id') + @ApiResponse({ status: 200, description: 'Détails du parent par ID utilisateur' }) + @ApiResponse({ status: 404, description: 'Parent non trouvé' }) + getOne(@Param('id') user_id: string): Promise { + return this.parentsService.findOne(user_id); + } - //Creation de parents - @Post() - @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) - @ApiBody({ type: CreateParentDto }) - @ApiResponse({ status: 201, description: 'Parent créé avec succès' }) - create(data: typeorm.DeepPartial): Promise { - const dto = data as CreateParentDto; - return this.parentsService.create({ ...dto, - user: { ...(dto as any).user, role: RoleType.PARENT } }); - } + @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) + @Post() + @ApiBody({ type: CreateParentDto }) + @ApiResponse({ status: 201, description: 'Parent créé avec succès' }) + create(@Body() dto: CreateParentDto): Promise { + return this.parentsService.create(dto); + } + @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) + @Patch(':id') + @ApiBody({ type: UpdateParentsDto }) + @ApiResponse({ status: 200, description: 'Parent mis à jour avec succès' }) + @ApiResponse({ status: 404, description: 'Parent introuvable' }) + update(@Param('id') id: string, @Body() dto: UpdateParentsDto): Promise { + return this.parentsService.update(id, dto); + } - - //Modifier un parent - @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) //Seul les utilisateurs super admin et gestionnaire peuvent accéder à cette route - @Patch(':id') - @ApiBody({ type: UpdateParentsDto }) - @ApiResponse({ status: 200, description: 'Parent mis a jour avec succès' }) - @ApiResponse({ status: 404, description: 'Parent introuvable' }) - update( - @Param('id') id: string, - @Body() dto: UpdateParentsDto - ): Promise { - return this.parentsService.update(id, dto); - } - - //Supprmer un parent - @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) //Seul les utilisateurs super admin et gestionnaire peuvent accéder à cette route - @Delete(':id') - @ApiResponse({ status: 200, description: 'Parent supprime avec succès' }) - @ApiResponse({ status: 404, description: 'Parent introuvable' }) - @ApiResponse({ status: 403, description: 'Acces refuse' }) - remove(@Param('id') id: string): Promise { - return this.parentsService.remove(id); - } + @Roles(RoleType.SUPER_ADMIN, RoleType.GESTIONNAIRE) + @Delete(':id') + @ApiResponse({ status: 200, description: 'Parent supprimé avec succès' }) + @ApiResponse({ status: 404, description: 'Parent introuvable' }) + @ApiResponse({ status: 403, description: 'Accès refusé' }) + remove(@Param('id') id: string): Promise { + return this.parentsService.remove(id); + } } diff --git a/src/routes/parents/parents.service.ts b/src/routes/parents/parents.service.ts index 9a0298b..d34e73d 100644 --- a/src/routes/parents/parents.service.ts +++ b/src/routes/parents/parents.service.ts @@ -1,69 +1,79 @@ -import { BadRequestException, ConflictException, Injectable, NotFoundException } from '@nestjs/common'; +import { + BadRequestException, + ConflictException, + Injectable, + NotFoundException, +} from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; -import { BaseService } from 'src/common/base.service'; +import { Repository } from 'typeorm'; import { Parents } from 'src/entities/parents.entity'; import { RoleType, Users } from 'src/entities/users.entity'; -import { DeepPartial, Repository } from 'typeorm'; import { CreateParentDto } from '../user/dto/create_parent.dto'; import { UpdateParentsDto } from '../user/dto/update_parent.dto'; @Injectable() -export class ParentsService extends BaseService { - constructor( - @InjectRepository(Parents) private readonly parentsRepository: Repository, - @InjectRepository(Users) private readonly usersRepository: Repository, - ) { - super(parentsRepository); +export class ParentsService { + constructor( + @InjectRepository(Parents) + private readonly parentsRepository: Repository, + @InjectRepository(Users) + private readonly usersRepository: Repository, + ) {} + + // Création d’un parent + async create(dto: CreateParentDto): Promise { + const user = await this.usersRepository.findOneBy({ id: dto.user_id }); + if (!user) throw new NotFoundException('Utilisateur introuvable'); + if (user.role !== RoleType.PARENT) { + throw new BadRequestException('Accès réservé aux parents'); } - override async create(data: DeepPartial): Promise { - const dto = data as CreateParentDto; - const user = await this.usersRepository.findOneBy({ id: dto.user_id }); - if (!user) throw new NotFoundException('Utilisateur introuvable'); - if (user.role !== RoleType.PARENT) throw new BadRequestException('Accès réservé aux parents'); + const exist = await this.parentsRepository.findOneBy({ user_id: dto.user_id }); + if (exist) throw new ConflictException('Ce parent existe déjà'); - const exist = await this.parentsRepository.findOneBy({ user_id: dto.user_id }); - if (exist) throw new ConflictException('Ce parent existe déjà'); - - let co_parent: Users | null = null; - if (dto.co_parent_id) { - co_parent = await this.usersRepository.findOneBy({ id: dto.co_parent_id }); - if (!co_parent) throw new NotFoundException('Co-parent introuvable'); - if (co_parent.role !== RoleType.PARENT) throw new BadRequestException('Accès réservé aux parents'); - } - - const entity = this.parentsRepository.create({ - user_id: dto.user_id, - user, - co_parent: co_parent ?? undefined, - }); - - return this.parentsRepository.save(entity); + let co_parent: Users | null = null; + if (dto.co_parent_id) { + co_parent = await this.usersRepository.findOneBy({ id: dto.co_parent_id }); + if (!co_parent) throw new NotFoundException('Co-parent introuvable'); + if (co_parent.role !== RoleType.PARENT) { + throw new BadRequestException('Accès réservé aux parents'); + } } + const entity = this.parentsRepository.create({ + user_id: dto.user_id, + user, + co_parent: co_parent ?? undefined, + }); - override async findAll(): Promise { - return this.parentsRepository.find({ - relations: ['user', 'co_parent', - 'parentChildren', 'dossiers'], - }); - } + return this.parentsRepository.save(entity); + } - override async findOne(user_id: string): Promise { - const parent = await this.parentsRepository.findOne({ - where: { user_id }, - relations: ['user', 'co_parent', 'parentChildren', 'dossiers'], - }); - if (!parent) throw new NotFoundException('Parent introuvable'); - return parent; - } + // Liste des parents + async findAll(): Promise { + return this.parentsRepository.find({ + relations: ['user', 'co_parent', 'parentChildren', 'dossiers'], + }); + } - async update(id: string, dto: UpdateParentsDto): Promise { - await this.parentsRepository.update(id, dto); - return this.findOne(id); - } + // Récupérer un parent par user_id + async findOne(user_id: string): Promise { + const parent = await this.parentsRepository.findOne({ + where: { user_id }, + relations: ['user', 'co_parent', 'parentChildren', 'dossiers'], + }); + if (!parent) throw new NotFoundException('Parent introuvable'); + return parent; + } - async remove(id: string): Promise { - await this.parentsRepository.delete(id); - } -} \ No newline at end of file + // Mise à jour + async update(id: string, dto: UpdateParentsDto): Promise { + await this.parentsRepository.update(id, dto); + return this.findOne(id); + } + + // Suppression + async remove(id: string): Promise { + await this.parentsRepository.delete(id); + } +}