From d8446080cce4020037c8cc934840321b0ab296a0 Mon Sep 17 00:00:00 2001 From: Valdior Date: Wed, 14 May 2025 21:45:16 +0200 Subject: [PATCH] init commit --- .eslintrc.js | 18 + .estlintignore | 5 + .gitignore | 21 + .prettierrc.js | 8 + README.md | 2 + __tests/accounts.playground.gql | 23 + __tests/admin/config.ts | 36 + __tests/admin/helpers/graphql-zeus.ts | 4164 +++++ __tests/admin/index.ts | 0 __tests/app/config.ts | 7 + __tests/app/helpers/graphql-zeus.ts | 6824 +++++++++ __tests/app/helpers/zeus/const.ts | 3093 ++++ __tests/app/helpers/zeus/index.ts | 3420 +++++ __tests/app/index.ts | 16 + __tests/app/src/bookings.ts | 72 + __tests/app/src/workspace.ts | 49 + __tests__/booking.gql | 11 + __tests__/signin.gql | 7 + __tests__/signup.gql | 2 + __tests__/workspace.gql | 66 + config/config.ts | 24 + config/domains.yaml | 19 + lib/__cronjobs/serverless-cron.yaml | 32 + lib/__hooks/index.ts | 29 + lib/seed/.gitignore | 19 + lib/seed/README.md | 73 + lib/seed/__loaders.ts | 46 + lib/seed/__tests/helper.ts | 33 + lib/seed/app-admin.ts | 66 + lib/seed/app.ts | 72 + lib/seed/devops/deploy-domains.js | 117 + lib/seed/devops/deploy-v2.js | 98 + lib/seed/devops/deploy.js | 103 + lib/seed/devops/handler-admin.ts | 10 + lib/seed/devops/handler-app.ts | 9 + lib/seed/devops/handler-hooks.ts | 153 + lib/seed/devops/serverless-admin.yaml | 55 + lib/seed/devops/serverless-app.yaml | 60 + lib/seed/devops/serverless-domains.yaml | 12 + lib/seed/devops/serverless-hooks.yaml | 37 + lib/seed/engine/EngineAccessService.ts | 161 + lib/seed/engine/EngineModel.ts | 277 + lib/seed/engine/EngineSchema.ts | 58 + lib/seed/engine/decorators/db.guard.ts | 40 + lib/seed/engine/decorators/pre.hooks.ts | 0 .../genericResolvers/BaseEngineResolver.ts | 114 + lib/seed/engine/utils/__interface.ts | 76 + lib/seed/engine/utils/crud.utils.ts | 721 + lib/seed/engine/utils/service.utils.ts | 118 + .../streams/schemas/stream.components.ts | 41 + .../streams/schemas/stream.schema.input.ts | 11 + .../utils/streams/schemas/stream.schema.ts | 50 + lib/seed/engine/utils/streams/stream.model.ts | 49 + .../engine/utils/streams/stream.resolver.ts | 0 .../engine/utils/streams/stream.service.ts | 84 + lib/seed/examples/__collections.ts | 9 + .../examples/__cronjobs/serverless-cron.yaml | 37 + .../__hooks/functions/example.hooks.ts | 8 + lib/seed/examples/__hooks/index.ts | 35 + lib/seed/examples/app-rest.ts | 11 + lib/seed/examples/config.ts | 34 + lib/seed/examples/domains.yaml | 17 + .../env/__env.accounts.createAdmin.json | 20 + .../examples/env/__env.firebase.import.json | 170 + .../examples/env/__env.general.import.json | 169 + .../env/__env.mailsettings.import.json | 219 + .../env/__env.notifications.import.json | 51 + lib/seed/examples/list.import.json | 47 + lib/seed/graphql/AccessService.ts | 150 + lib/seed/graphql/BaseGraphModel.ts | 828 + lib/seed/graphql/BaseService.ts | 122 + lib/seed/graphql/Middleware.ts | 204 + lib/seed/graphql/MiddlewareV2.ts | 132 + lib/seed/graphql/Request.ts | 54 + lib/seed/graphql/Server.ts | 102 + lib/seed/graphql/Settings.ts | 202 + .../graphql/baseModels/BaseContentModel.ts | 106 + lib/seed/graphql/baseModels/BaseSeoModel.ts | 103 + .../graphql/baseModels/BaseSeoSimpleModel.ts | 107 + .../baseResolvers/BasePublicResolver.ts | 143 + .../graphql/baseResolvers/BaseResolver.ts | 100 + .../genericResolvers/BaseGenericResolver.ts | 97 + .../graphql/genericResolvers/BaseResolver.ts | 100 + lib/seed/helpers/Context.ts | 12 + lib/seed/helpers/Error.ts | 43 + lib/seed/helpers/FetchHelper.ts | 64 + lib/seed/helpers/Request.ts | 29 + lib/seed/helpers/StringHelper.ts | 28 + lib/seed/helpers/TestsHelper.ts | 44 + lib/seed/helpers/Utils.checks.ts | 14 + lib/seed/helpers/Utils.dates.intervals.ts | 88 + lib/seed/helpers/Utils.dates.ts | 310 + lib/seed/helpers/Utils.string.ts | 38 + lib/seed/helpers/Utils.ts | 64 + lib/seed/interfaces/components.dates.ts | 114 + lib/seed/interfaces/components.errors.ts | 99 + lib/seed/interfaces/components.geo.ts | 192 + .../interfaces/components.notifications.ts | 24 + lib/seed/interfaces/components.texts.ts | 46 + lib/seed/interfaces/components.ts | 318 + lib/seed/interfaces/context.ts | 22 + lib/seed/interfaces/permission.ts | 24 + lib/seed/interfaces/response.ts | 28 + lib/seed/services/auth/FirebaseAuthService.ts | 147 + lib/seed/services/auth/FirebaseService.ts | 211 + .../change-stream/change-stream.components.ts | 66 + .../change-stream/change-stream.model.ts | 68 + .../change-stream/change-stream.resolver.ts | 27 + .../change-stream/change-stream.service.ts | 0 .../services/database/DBRequestService.ts | 561 + lib/seed/services/database/DBService.ts | 61 + .../services/database/FirestoreService.ts | 67 + lib/seed/services/database/LoaderService.ts | 59 + lib/seed/services/email/EmailHelper.ts | 118 + lib/seed/services/email/EmailService.ts | 20 + lib/seed/services/geolocation/README.md | 9 + .../__tests/geolocation.playground.gql | 0 .../geolocation/components/countries.ts | 514 + .../functions/GooglePlaceResolver.ts | 59 + .../geolocation/models/BaseGeoSeoModel.ts | 66 + .../geolocation/models/BaseGeoSimpleModel.ts | 83 + .../geolocation/services/GeolocService.ts | 9 + .../services/GooglePlaceService.ts | 173 + lib/seed/services/hooks/hooks.decorator.ts | 126 + .../helpers/notification.helper.ts | 59 + .../notifications/helpers/slack.helper.ts | 60 + .../notifications/helpers/sms.helper.ts | 63 + .../notifications/notifications.model.ts | 41 + .../schemas/notifications.schema.ts | 82 + .../services/NotificationService.ts | 93 + lib/seed/webpack.config.js | 50 + package.json | 121 + serverless.yaml | 44 + services/api-messaging/README.md | 6 + services/api-messaging/__tests/playground.gql | 67 + services/api-messaging/app.ts | 74 + services/api-messaging/components.ts | 35 + .../functions/messages/message.model.ts | 91 + .../functions/messages/message.resolver.ts | 161 + .../functions/rooms/room.model.ts | 115 + .../functions/rooms/room.resolver.ts | 211 + .../functions/rooms/room.service.ts | 17 + .../api-messaging/serverless-messaging.yaml | 40 + services/api-uploads/README.md | 9 + services/api-uploads/__tests/playground.gql | 25 + services/api-uploads/app.ts | 79 + services/api-uploads/file.model.ts | 92 + services/api-uploads/file.resolver.ts | 59 + services/api-uploads/serverless-upload.yaml | 40 + services/api-uploads/services/ImageService.ts | 53 + services/api-uploads/services/S3Service.ts | 57 + .../api-uploads/services/UploadService.ts | 133 + services/module-accounts/.gitignore | 2 + services/module-accounts/README.md | 19 + .../module-accounts/__tests/accounts.test.ts | 74 + .../__tests/admin.playground.graphql | 5 + .../__tests/playground.graphql | 45 + .../module-accounts/account.components.ts | 140 + services/module-accounts/account.model.ts | 258 + .../module-accounts/account.resolver.admin.ts | 125 + services/module-accounts/account.resolver.ts | 171 + .../env/emails.settings.import.json | 229 + .../env/env.settings.import.json | 51 + .../functions/auth-history/pages.model.ts | 51 + .../auth-history/pages.resolver.admin.ts | 72 + .../schemas/auth-history.schema.ts | 45 + .../auth-history/schemas/pages.input.ts | 16 + .../auth/account.auth.mfa.resolver.ts | 246 + .../resolvers/auth/account.auth.resolver.ts | 271 + .../services/AccountService.ts | 72 + .../services/account.service.helper.ts | 136 + services/module-booking/.gitignore | 2 + services/module-booking/README.md | 23 + .../module-booking/__tests/playground.gql | 11 + services/module-booking/components.errors.ts | 6 + .../functions/bookings/booking.service.ts | 156 + .../bookings/bookings.engine.model.ts | 71 + .../resolvers/bookings.engine.resolver.ts | 42 + .../bookings/schemas/bookings.engine.input.ts | 43 + .../schemas/bookings.engine.schema.ts | 76 + .../module-booking/functions/components.ts | 44 + .../functions/crons/booking.cron.service.ts | 31 + services/module-booking/functions/helper.ts | 63 + services/module-cms/.gitignore | 2 + services/module-cms/README.md | 7 + .../__tests/pages.playground.graphql | 31 + .../project-settings.playground.graphql | 1 + services/module-cms/components.ts | 33 + .../functions/categories/category.model.ts | 80 + .../categories/category.resolver.admin.ts | 28 + .../categories/category.resolver.fields.ts | 15 + .../functions/categories/category.resolver.ts | 26 + .../functions/countries/countries.json | 1254 ++ .../functions/countries/country.components.ts | 177 + .../functions/countries/country.model.ts | 41 + .../functions/countries/country.service.ts | 28 + .../resolvers/country.resolver.admin.ts | 73 + .../countries/resolvers/country.resolver.ts | 44 + .../countries/schemas/country.input.ts | 34 + .../countries/schemas/country.schema.ts | 34 + .../emails-settings.components.ts | 57 + .../emails-settings/emails-settings.model.ts | 191 + .../emails-settings.resolver.ts | 33 + .../errors-settings.components.ts | 0 .../error-settings/errors-settings.model.ts | 91 + .../errors-settings.resolver.ts | 33 + .../module-cms/functions/lists/list.model.ts | 75 + .../functions/lists/list.resolver.admin.ts | 26 + .../functions/lists/list.resolver.fields.ts | 15 + .../functions/lists/list.resolver.ts | 26 + .../module-cms/functions/pages/pages.model.ts | 51 + .../functions/pages/pages.resolver.admin.ts | 72 + .../functions/pages/schemas/pages.input.ts | 16 + .../functions/pages/schemas/pages.schema.ts | 38 + .../project-settings.components.ts | 0 .../project-settings.model.ts | 71 + .../project-settings.resolver.admin.ts | 35 + .../module-cms/import/pages.countries.json | 12690 ++++++++++++++++ services/module-cms/index.ts | 19 + services/module-cms/services/CmsService.ts | 66 + services/module-cms/services/PageService.ts | 32 + .../module-cms/services/TreatmentContent.ts | 172 + services/module-comments/.gitignore | 2 + services/module-comments/README.md | 14 + .../module-comments/__tests/playground.gql | 13 + .../functions/comments/comment.model.ts | 47 + .../functions/comments/comment.service.ts | 111 + .../functions/comments/commentStat.model.ts | 60 + .../comments/resolvers/comment.resolver.ts | 53 + .../comments/schemas/comments.schema.input.ts | 34 + .../comments/schemas/comments.schema.ts | 25 + .../module-comments/functions/components.ts | 83 + .../reviews/resolvers/reviews.resolver.ts | 53 + .../functions/reviews/reviews.model.ts | 52 + .../reviews/schemas/reviews.schema.input.ts | 36 + .../reviews/schemas/reviews.schema.ts | 23 + services/module-favorites/.gitignore | 2 + services/module-favorites/README.md | 34 + .../__tests/favorites.playground.graphql | 7 + services/module-favorites/components.ts | 54 + .../favLikes/favLikes.resolver.admin.ts | 14 + .../functions/favLikes/favLikes.resolver.ts | 211 + services/module-payments/.gitignore | 2 + services/module-payments/README.md | 28 + .../__examples/checkout.resolver.ts | 46 + .../__examples/env.import.json | 102 + .../__examples/stripe.env.json | 85 + .../__handlers/stripeHookHandler.ts | 102 + .../__tests/billingInfo.playground.graphql | 30 + .../__tests/pm.playground.graphql | 18 + .../components/components.errors.ts | 8 + .../components/components.orders.ts | 85 + .../components/components.payments.ts | 205 + .../module-payments/components/components.ts | 62 + .../components/stripe.components.ts | 48 + .../billingInfo.resolver.admin.ts | 14 + .../billingInfo.resolver.fields.ts | 14 + .../billingInfos/billingInfo.resolver.ts | 75 + .../functions/orders/__index.ts | 10 + .../functions/orders/carts/__index.ts | 6 + .../orders/carts/resolvers/cart.resolver.ts | 274 + .../carts/schemas/cart.schema.inputs.ts | 30 + .../orders/carts/schemas/line.schema.ts | 134 + .../functions/orders/helpers/OrderHelper.ts | 35 + .../functions/orders/order.model.ts | 60 + .../orders/resolvers/admin/admin.mutations.ts | 116 + .../orders/resolvers/admin/admin.queries.ts | 46 + .../orders/resolvers/app/app.resolver.ts | 86 + .../orders/resolvers/app/checkout.resolver.ts | 54 + .../resolvers/business/business.input.ts | 7 + .../resolvers/business/business.mutations.ts | 58 + .../resolvers/business/business.queries.ts | 67 + .../resolvers/business/business.service.ts | 23 + .../orders/resolvers/field.resolver.ts | 9 + .../orders/schemas/order.schema.input.ts | 49 + .../functions/orders/schemas/order.schema.ts | 140 + .../orders/schemas/provider.schema.ts | 19 + .../functions/orders/services/CartService.ts | 113 + .../functions/orders/services/LineService.ts | 0 .../functions/orders/services/OrderService.ts | 290 + .../paymentMethods/pm.resolver.admin.ts | 14 + .../paymentMethods/pm.resolver.fields.ts | 14 + .../functions/paymentMethods/pm.resolver.ts | 95 + .../payoutAccounts/pa.resolver.admin.ts | 14 + .../payoutAccounts/pa.resolver.fields.ts | 14 + .../functions/payoutAccounts/pa.resolver.ts | 61 + .../functions/promos/component.ts | 9 + .../functions/promos/promo.model.ts | 171 + .../functions/promos/promo.resolver.admin.ts | 51 + .../functions/promos/promo.resolver.ts | 32 + .../functions/vat/vat.components.ts | 761 + .../functions/vat/vat.model.ts | 104 + .../functions/vat/vat.resolver.admin.ts | 71 + .../functions/vat/vat.resolver.ts | 89 + .../helpers/payments.helpers.ts | 93 + services/module-payments/index.ts | 9 + .../module-payments/serverless-payments.yaml | 46 + .../services/stripe/StripeService.ts | 382 + .../services/stripe/paymentIntent.helper.ts | 14 + .../services/stripe/payoutAccounts.helper.ts | 0 src/__components/components.ts | 157 + src/__indexes/__bootstrap.ts | 29 + src/__indexes/__collections.ts | 11 + src/__indexes/__loaders.ts | 68 + src/__indexes/__resolvers.index.admin.ts | 25 + src/__indexes/__resolvers.index.ts | 50 + src/accounts/account.components.ts | 152 + src/accounts/account.helper.ts | 3 + src/accounts/account.model.ts | 77 + src/accounts/account.resolver.ts | 57 + src/accounts/account.service.ts | 0 src/bookings/resolvers/bookings.resolver.ts | 140 + src/bookings/resolvers/getEventDescription.ts | 18 + src/bookings/schemas/bookings.input.ts | 20 + src/bookings/schemas/bookings.schema.ts | 22 + src/orders/components/checkout.schema.ts | 6 + src/orders/components/components.cart.ts | 10 + src/orders/components/line.schema.ts | 44 + src/orders/order.model.ts | 23 + src/orders/order.resolver.admin.ts | 0 src/orders/order.resolver.ts | 55 + src/orders/services/line.service.ts | 56 + src/orders/services/order.service.ts | 186 + src/workspace/workspace.model.ts | 288 + src/workspace/workspace.resolver.admin.ts | 29 + src/workspace/workspace.resolver.fields.ts | 62 + src/workspace/workspace.resolver.ts | 102 + tsconfig.json | 34 + 328 files changed, 55669 insertions(+) create mode 100644 .eslintrc.js create mode 100644 .estlintignore create mode 100644 .gitignore create mode 100644 .prettierrc.js create mode 100644 README.md create mode 100644 __tests/accounts.playground.gql create mode 100644 __tests/admin/config.ts create mode 100644 __tests/admin/helpers/graphql-zeus.ts create mode 100644 __tests/admin/index.ts create mode 100644 __tests/app/config.ts create mode 100644 __tests/app/helpers/graphql-zeus.ts create mode 100644 __tests/app/helpers/zeus/const.ts create mode 100644 __tests/app/helpers/zeus/index.ts create mode 100644 __tests/app/index.ts create mode 100644 __tests/app/src/bookings.ts create mode 100644 __tests/app/src/workspace.ts create mode 100644 __tests__/booking.gql create mode 100644 __tests__/signin.gql create mode 100644 __tests__/signup.gql create mode 100644 __tests__/workspace.gql create mode 100644 config/config.ts create mode 100644 config/domains.yaml create mode 100644 lib/__cronjobs/serverless-cron.yaml create mode 100644 lib/__hooks/index.ts create mode 100644 lib/seed/.gitignore create mode 100644 lib/seed/README.md create mode 100644 lib/seed/__loaders.ts create mode 100644 lib/seed/__tests/helper.ts create mode 100644 lib/seed/app-admin.ts create mode 100644 lib/seed/app.ts create mode 100644 lib/seed/devops/deploy-domains.js create mode 100644 lib/seed/devops/deploy-v2.js create mode 100644 lib/seed/devops/deploy.js create mode 100644 lib/seed/devops/handler-admin.ts create mode 100644 lib/seed/devops/handler-app.ts create mode 100644 lib/seed/devops/handler-hooks.ts create mode 100644 lib/seed/devops/serverless-admin.yaml create mode 100644 lib/seed/devops/serverless-app.yaml create mode 100644 lib/seed/devops/serverless-domains.yaml create mode 100644 lib/seed/devops/serverless-hooks.yaml create mode 100644 lib/seed/engine/EngineAccessService.ts create mode 100644 lib/seed/engine/EngineModel.ts create mode 100644 lib/seed/engine/EngineSchema.ts create mode 100644 lib/seed/engine/decorators/db.guard.ts create mode 100644 lib/seed/engine/decorators/pre.hooks.ts create mode 100644 lib/seed/engine/genericResolvers/BaseEngineResolver.ts create mode 100644 lib/seed/engine/utils/__interface.ts create mode 100644 lib/seed/engine/utils/crud.utils.ts create mode 100644 lib/seed/engine/utils/service.utils.ts create mode 100644 lib/seed/engine/utils/streams/schemas/stream.components.ts create mode 100644 lib/seed/engine/utils/streams/schemas/stream.schema.input.ts create mode 100644 lib/seed/engine/utils/streams/schemas/stream.schema.ts create mode 100644 lib/seed/engine/utils/streams/stream.model.ts create mode 100644 lib/seed/engine/utils/streams/stream.resolver.ts create mode 100644 lib/seed/engine/utils/streams/stream.service.ts create mode 100644 lib/seed/examples/__collections.ts create mode 100644 lib/seed/examples/__cronjobs/serverless-cron.yaml create mode 100644 lib/seed/examples/__hooks/functions/example.hooks.ts create mode 100644 lib/seed/examples/__hooks/index.ts create mode 100644 lib/seed/examples/app-rest.ts create mode 100644 lib/seed/examples/config.ts create mode 100644 lib/seed/examples/domains.yaml create mode 100644 lib/seed/examples/env/__env.accounts.createAdmin.json create mode 100644 lib/seed/examples/env/__env.firebase.import.json create mode 100644 lib/seed/examples/env/__env.general.import.json create mode 100644 lib/seed/examples/env/__env.mailsettings.import.json create mode 100644 lib/seed/examples/env/__env.notifications.import.json create mode 100644 lib/seed/examples/list.import.json create mode 100644 lib/seed/graphql/AccessService.ts create mode 100644 lib/seed/graphql/BaseGraphModel.ts create mode 100644 lib/seed/graphql/BaseService.ts create mode 100644 lib/seed/graphql/Middleware.ts create mode 100644 lib/seed/graphql/MiddlewareV2.ts create mode 100644 lib/seed/graphql/Request.ts create mode 100644 lib/seed/graphql/Server.ts create mode 100644 lib/seed/graphql/Settings.ts create mode 100644 lib/seed/graphql/baseModels/BaseContentModel.ts create mode 100644 lib/seed/graphql/baseModels/BaseSeoModel.ts create mode 100644 lib/seed/graphql/baseModels/BaseSeoSimpleModel.ts create mode 100644 lib/seed/graphql/baseResolvers/BasePublicResolver.ts create mode 100644 lib/seed/graphql/baseResolvers/BaseResolver.ts create mode 100644 lib/seed/graphql/genericResolvers/BaseGenericResolver.ts create mode 100644 lib/seed/graphql/genericResolvers/BaseResolver.ts create mode 100644 lib/seed/helpers/Context.ts create mode 100644 lib/seed/helpers/Error.ts create mode 100644 lib/seed/helpers/FetchHelper.ts create mode 100644 lib/seed/helpers/Request.ts create mode 100644 lib/seed/helpers/StringHelper.ts create mode 100644 lib/seed/helpers/TestsHelper.ts create mode 100644 lib/seed/helpers/Utils.checks.ts create mode 100644 lib/seed/helpers/Utils.dates.intervals.ts create mode 100644 lib/seed/helpers/Utils.dates.ts create mode 100644 lib/seed/helpers/Utils.string.ts create mode 100644 lib/seed/helpers/Utils.ts create mode 100644 lib/seed/interfaces/components.dates.ts create mode 100644 lib/seed/interfaces/components.errors.ts create mode 100644 lib/seed/interfaces/components.geo.ts create mode 100644 lib/seed/interfaces/components.notifications.ts create mode 100644 lib/seed/interfaces/components.texts.ts create mode 100644 lib/seed/interfaces/components.ts create mode 100644 lib/seed/interfaces/context.ts create mode 100644 lib/seed/interfaces/permission.ts create mode 100644 lib/seed/interfaces/response.ts create mode 100644 lib/seed/services/auth/FirebaseAuthService.ts create mode 100644 lib/seed/services/auth/FirebaseService.ts create mode 100644 lib/seed/services/change-stream/change-stream.components.ts create mode 100644 lib/seed/services/change-stream/change-stream.model.ts create mode 100644 lib/seed/services/change-stream/change-stream.resolver.ts create mode 100644 lib/seed/services/change-stream/change-stream.service.ts create mode 100644 lib/seed/services/database/DBRequestService.ts create mode 100644 lib/seed/services/database/DBService.ts create mode 100644 lib/seed/services/database/FirestoreService.ts create mode 100644 lib/seed/services/database/LoaderService.ts create mode 100644 lib/seed/services/email/EmailHelper.ts create mode 100644 lib/seed/services/email/EmailService.ts create mode 100644 lib/seed/services/geolocation/README.md create mode 100644 lib/seed/services/geolocation/__tests/geolocation.playground.gql create mode 100644 lib/seed/services/geolocation/components/countries.ts create mode 100644 lib/seed/services/geolocation/functions/GooglePlaceResolver.ts create mode 100644 lib/seed/services/geolocation/models/BaseGeoSeoModel.ts create mode 100644 lib/seed/services/geolocation/models/BaseGeoSimpleModel.ts create mode 100644 lib/seed/services/geolocation/services/GeolocService.ts create mode 100644 lib/seed/services/geolocation/services/GooglePlaceService.ts create mode 100644 lib/seed/services/hooks/hooks.decorator.ts create mode 100644 lib/seed/services/notifications/helpers/notification.helper.ts create mode 100644 lib/seed/services/notifications/helpers/slack.helper.ts create mode 100644 lib/seed/services/notifications/helpers/sms.helper.ts create mode 100644 lib/seed/services/notifications/notifications.model.ts create mode 100644 lib/seed/services/notifications/schemas/notifications.schema.ts create mode 100644 lib/seed/services/notifications/services/NotificationService.ts create mode 100644 lib/seed/webpack.config.js create mode 100644 package.json create mode 100644 serverless.yaml create mode 100644 services/api-messaging/README.md create mode 100644 services/api-messaging/__tests/playground.gql create mode 100644 services/api-messaging/app.ts create mode 100644 services/api-messaging/components.ts create mode 100644 services/api-messaging/functions/messages/message.model.ts create mode 100644 services/api-messaging/functions/messages/message.resolver.ts create mode 100644 services/api-messaging/functions/rooms/room.model.ts create mode 100644 services/api-messaging/functions/rooms/room.resolver.ts create mode 100644 services/api-messaging/functions/rooms/room.service.ts create mode 100644 services/api-messaging/serverless-messaging.yaml create mode 100644 services/api-uploads/README.md create mode 100644 services/api-uploads/__tests/playground.gql create mode 100644 services/api-uploads/app.ts create mode 100644 services/api-uploads/file.model.ts create mode 100644 services/api-uploads/file.resolver.ts create mode 100644 services/api-uploads/serverless-upload.yaml create mode 100644 services/api-uploads/services/ImageService.ts create mode 100644 services/api-uploads/services/S3Service.ts create mode 100644 services/api-uploads/services/UploadService.ts create mode 100644 services/module-accounts/.gitignore create mode 100644 services/module-accounts/README.md create mode 100644 services/module-accounts/__tests/accounts.test.ts create mode 100644 services/module-accounts/__tests/admin.playground.graphql create mode 100644 services/module-accounts/__tests/playground.graphql create mode 100644 services/module-accounts/account.components.ts create mode 100644 services/module-accounts/account.model.ts create mode 100644 services/module-accounts/account.resolver.admin.ts create mode 100644 services/module-accounts/account.resolver.ts create mode 100644 services/module-accounts/env/emails.settings.import.json create mode 100644 services/module-accounts/env/env.settings.import.json create mode 100644 services/module-accounts/functions/auth-history/pages.model.ts create mode 100644 services/module-accounts/functions/auth-history/pages.resolver.admin.ts create mode 100644 services/module-accounts/functions/auth-history/schemas/auth-history.schema.ts create mode 100644 services/module-accounts/functions/auth-history/schemas/pages.input.ts create mode 100644 services/module-accounts/resolvers/auth/account.auth.mfa.resolver.ts create mode 100644 services/module-accounts/resolvers/auth/account.auth.resolver.ts create mode 100644 services/module-accounts/services/AccountService.ts create mode 100644 services/module-accounts/services/account.service.helper.ts create mode 100644 services/module-booking/.gitignore create mode 100644 services/module-booking/README.md create mode 100644 services/module-booking/__tests/playground.gql create mode 100644 services/module-booking/components.errors.ts create mode 100644 services/module-booking/functions/bookings/booking.service.ts create mode 100644 services/module-booking/functions/bookings/bookings.engine.model.ts create mode 100644 services/module-booking/functions/bookings/resolvers/bookings.engine.resolver.ts create mode 100644 services/module-booking/functions/bookings/schemas/bookings.engine.input.ts create mode 100644 services/module-booking/functions/bookings/schemas/bookings.engine.schema.ts create mode 100644 services/module-booking/functions/components.ts create mode 100644 services/module-booking/functions/crons/booking.cron.service.ts create mode 100644 services/module-booking/functions/helper.ts create mode 100644 services/module-cms/.gitignore create mode 100644 services/module-cms/README.md create mode 100644 services/module-cms/__tests/pages.playground.graphql create mode 100644 services/module-cms/__tests/project-settings.playground.graphql create mode 100644 services/module-cms/components.ts create mode 100644 services/module-cms/functions/categories/category.model.ts create mode 100644 services/module-cms/functions/categories/category.resolver.admin.ts create mode 100644 services/module-cms/functions/categories/category.resolver.fields.ts create mode 100644 services/module-cms/functions/categories/category.resolver.ts create mode 100644 services/module-cms/functions/countries/countries.json create mode 100644 services/module-cms/functions/countries/country.components.ts create mode 100644 services/module-cms/functions/countries/country.model.ts create mode 100644 services/module-cms/functions/countries/country.service.ts create mode 100644 services/module-cms/functions/countries/resolvers/country.resolver.admin.ts create mode 100644 services/module-cms/functions/countries/resolvers/country.resolver.ts create mode 100644 services/module-cms/functions/countries/schemas/country.input.ts create mode 100644 services/module-cms/functions/countries/schemas/country.schema.ts create mode 100644 services/module-cms/functions/emails-settings/emails-settings.components.ts create mode 100644 services/module-cms/functions/emails-settings/emails-settings.model.ts create mode 100644 services/module-cms/functions/emails-settings/emails-settings.resolver.ts create mode 100644 services/module-cms/functions/error-settings/errors-settings.components.ts create mode 100644 services/module-cms/functions/error-settings/errors-settings.model.ts create mode 100644 services/module-cms/functions/error-settings/errors-settings.resolver.ts create mode 100644 services/module-cms/functions/lists/list.model.ts create mode 100644 services/module-cms/functions/lists/list.resolver.admin.ts create mode 100644 services/module-cms/functions/lists/list.resolver.fields.ts create mode 100644 services/module-cms/functions/lists/list.resolver.ts create mode 100644 services/module-cms/functions/pages/pages.model.ts create mode 100644 services/module-cms/functions/pages/pages.resolver.admin.ts create mode 100644 services/module-cms/functions/pages/schemas/pages.input.ts create mode 100644 services/module-cms/functions/pages/schemas/pages.schema.ts create mode 100644 services/module-cms/functions/project-settings/project-settings.components.ts create mode 100644 services/module-cms/functions/project-settings/project-settings.model.ts create mode 100644 services/module-cms/functions/project-settings/project-settings.resolver.admin.ts create mode 100644 services/module-cms/import/pages.countries.json create mode 100644 services/module-cms/index.ts create mode 100644 services/module-cms/services/CmsService.ts create mode 100644 services/module-cms/services/PageService.ts create mode 100644 services/module-cms/services/TreatmentContent.ts create mode 100644 services/module-comments/.gitignore create mode 100644 services/module-comments/README.md create mode 100644 services/module-comments/__tests/playground.gql create mode 100644 services/module-comments/functions/comments/comment.model.ts create mode 100644 services/module-comments/functions/comments/comment.service.ts create mode 100644 services/module-comments/functions/comments/commentStat.model.ts create mode 100644 services/module-comments/functions/comments/resolvers/comment.resolver.ts create mode 100644 services/module-comments/functions/comments/schemas/comments.schema.input.ts create mode 100644 services/module-comments/functions/comments/schemas/comments.schema.ts create mode 100644 services/module-comments/functions/components.ts create mode 100644 services/module-comments/functions/reviews/resolvers/reviews.resolver.ts create mode 100644 services/module-comments/functions/reviews/reviews.model.ts create mode 100644 services/module-comments/functions/reviews/schemas/reviews.schema.input.ts create mode 100644 services/module-comments/functions/reviews/schemas/reviews.schema.ts create mode 100644 services/module-favorites/.gitignore create mode 100644 services/module-favorites/README.md create mode 100644 services/module-favorites/__tests/favorites.playground.graphql create mode 100644 services/module-favorites/components.ts create mode 100644 services/module-favorites/functions/favLikes/favLikes.resolver.admin.ts create mode 100644 services/module-favorites/functions/favLikes/favLikes.resolver.ts create mode 100644 services/module-payments/.gitignore create mode 100644 services/module-payments/README.md create mode 100644 services/module-payments/__examples/checkout.resolver.ts create mode 100644 services/module-payments/__examples/env.import.json create mode 100644 services/module-payments/__examples/stripe.env.json create mode 100644 services/module-payments/__handlers/stripeHookHandler.ts create mode 100644 services/module-payments/__tests/billingInfo.playground.graphql create mode 100644 services/module-payments/__tests/pm.playground.graphql create mode 100644 services/module-payments/components/components.errors.ts create mode 100644 services/module-payments/components/components.orders.ts create mode 100644 services/module-payments/components/components.payments.ts create mode 100644 services/module-payments/components/components.ts create mode 100644 services/module-payments/components/stripe.components.ts create mode 100644 services/module-payments/functions/billingInfos/billingInfo.resolver.admin.ts create mode 100644 services/module-payments/functions/billingInfos/billingInfo.resolver.fields.ts create mode 100644 services/module-payments/functions/billingInfos/billingInfo.resolver.ts create mode 100644 services/module-payments/functions/orders/__index.ts create mode 100644 services/module-payments/functions/orders/carts/__index.ts create mode 100644 services/module-payments/functions/orders/carts/resolvers/cart.resolver.ts create mode 100644 services/module-payments/functions/orders/carts/schemas/cart.schema.inputs.ts create mode 100644 services/module-payments/functions/orders/carts/schemas/line.schema.ts create mode 100644 services/module-payments/functions/orders/helpers/OrderHelper.ts create mode 100644 services/module-payments/functions/orders/order.model.ts create mode 100644 services/module-payments/functions/orders/resolvers/admin/admin.mutations.ts create mode 100644 services/module-payments/functions/orders/resolvers/admin/admin.queries.ts create mode 100644 services/module-payments/functions/orders/resolvers/app/app.resolver.ts create mode 100644 services/module-payments/functions/orders/resolvers/app/checkout.resolver.ts create mode 100644 services/module-payments/functions/orders/resolvers/business/business.input.ts create mode 100644 services/module-payments/functions/orders/resolvers/business/business.mutations.ts create mode 100644 services/module-payments/functions/orders/resolvers/business/business.queries.ts create mode 100644 services/module-payments/functions/orders/resolvers/business/business.service.ts create mode 100644 services/module-payments/functions/orders/resolvers/field.resolver.ts create mode 100644 services/module-payments/functions/orders/schemas/order.schema.input.ts create mode 100644 services/module-payments/functions/orders/schemas/order.schema.ts create mode 100644 services/module-payments/functions/orders/schemas/provider.schema.ts create mode 100644 services/module-payments/functions/orders/services/CartService.ts create mode 100644 services/module-payments/functions/orders/services/LineService.ts create mode 100644 services/module-payments/functions/orders/services/OrderService.ts create mode 100644 services/module-payments/functions/paymentMethods/pm.resolver.admin.ts create mode 100644 services/module-payments/functions/paymentMethods/pm.resolver.fields.ts create mode 100644 services/module-payments/functions/paymentMethods/pm.resolver.ts create mode 100644 services/module-payments/functions/payoutAccounts/pa.resolver.admin.ts create mode 100644 services/module-payments/functions/payoutAccounts/pa.resolver.fields.ts create mode 100644 services/module-payments/functions/payoutAccounts/pa.resolver.ts create mode 100644 services/module-payments/functions/promos/component.ts create mode 100644 services/module-payments/functions/promos/promo.model.ts create mode 100644 services/module-payments/functions/promos/promo.resolver.admin.ts create mode 100644 services/module-payments/functions/promos/promo.resolver.ts create mode 100644 services/module-payments/functions/vat/vat.components.ts create mode 100644 services/module-payments/functions/vat/vat.model.ts create mode 100644 services/module-payments/functions/vat/vat.resolver.admin.ts create mode 100644 services/module-payments/functions/vat/vat.resolver.ts create mode 100644 services/module-payments/helpers/payments.helpers.ts create mode 100644 services/module-payments/index.ts create mode 100644 services/module-payments/serverless-payments.yaml create mode 100644 services/module-payments/services/stripe/StripeService.ts create mode 100644 services/module-payments/services/stripe/paymentIntent.helper.ts create mode 100644 services/module-payments/services/stripe/payoutAccounts.helper.ts create mode 100644 src/__components/components.ts create mode 100644 src/__indexes/__bootstrap.ts create mode 100644 src/__indexes/__collections.ts create mode 100644 src/__indexes/__loaders.ts create mode 100644 src/__indexes/__resolvers.index.admin.ts create mode 100644 src/__indexes/__resolvers.index.ts create mode 100644 src/accounts/account.components.ts create mode 100644 src/accounts/account.helper.ts create mode 100644 src/accounts/account.model.ts create mode 100644 src/accounts/account.resolver.ts create mode 100644 src/accounts/account.service.ts create mode 100644 src/bookings/resolvers/bookings.resolver.ts create mode 100644 src/bookings/resolvers/getEventDescription.ts create mode 100644 src/bookings/schemas/bookings.input.ts create mode 100644 src/bookings/schemas/bookings.schema.ts create mode 100644 src/orders/components/checkout.schema.ts create mode 100644 src/orders/components/components.cart.ts create mode 100644 src/orders/components/line.schema.ts create mode 100644 src/orders/order.model.ts create mode 100644 src/orders/order.resolver.admin.ts create mode 100644 src/orders/order.resolver.ts create mode 100644 src/orders/services/line.service.ts create mode 100644 src/orders/services/order.service.ts create mode 100644 src/workspace/workspace.model.ts create mode 100644 src/workspace/workspace.resolver.admin.ts create mode 100644 src/workspace/workspace.resolver.fields.ts create mode 100644 src/workspace/workspace.resolver.ts create mode 100644 tsconfig.json diff --git a/.eslintrc.js b/.eslintrc.js new file mode 100644 index 0000000..50092b9 --- /dev/null +++ b/.eslintrc.js @@ -0,0 +1,18 @@ +module.exports = { + parser: '@typescript-eslint/parser', // Specifies the ESLint parser + extends: [ + 'plugin:@typescript-eslint/recommended', // Uses the recommended rules from the @typescript-eslint/eslint-plugin + 'prettier/@typescript-eslint', // Uses eslint-config-prettier to disable ESLint rules from @typescript-eslint/eslint-plugin that would conflict with prettier + 'plugin:prettier/recommended', // Enables eslint-plugin-prettier and displays prettier errors as ESLint errors. Make sure this is always the last configuration in the extends array. + ], + parserOptions: { + ecmaVersion: 2018, // Allows for the parsing of modern ECMAScript features + sourceType: 'module', // Allows for the use of imports + }, + rules: { + // Place to specify ESLint rules. Can be used to overwrite rules specified from the extended configs + // e.g. "@typescript-eslint/explicit-function-return-type": "off", + "@typescript-eslint/no-explicit-any":"off", + "@typescript-eslint/interface-name-prefix":"off" + }, +}; \ No newline at end of file diff --git a/.estlintignore b/.estlintignore new file mode 100644 index 0000000..5d64e8f --- /dev/null +++ b/.estlintignore @@ -0,0 +1,5 @@ +.webpack/* + +schema.d.ts +validation.ts +redoc-static.html \ No newline at end of file diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..89a5c55 --- /dev/null +++ b/.gitignore @@ -0,0 +1,21 @@ +node_modules +.DS_Store +package-lock.json + +.build +.webpack + +*/**/.env +*/**/postman.json +*/**/schema.d.ts +*/**/*.d.ts + +src/config/.env.json + +.serverless/ + +config/.env.json + +handler.ts +.slslogs.txt +.vscode \ No newline at end of file diff --git a/.prettierrc.js b/.prettierrc.js new file mode 100644 index 0000000..8aa1879 --- /dev/null +++ b/.prettierrc.js @@ -0,0 +1,8 @@ +module.exports = { + semi: true, + trailingComma: 'all', + singleQuote: true, + printWidth: 150, + tabWidth: 4, + arrowParens: "always" +}; \ No newline at end of file diff --git a/README.md b/README.md new file mode 100644 index 0000000..ce9ad5b --- /dev/null +++ b/README.md @@ -0,0 +1,2 @@ +# workinflex-backend + diff --git a/__tests/accounts.playground.gql b/__tests/accounts.playground.gql new file mode 100644 index 0000000..ae48b65 --- /dev/null +++ b/__tests/accounts.playground.gql @@ -0,0 +1,23 @@ +mutation getCode { + signInEmailCodeRequest(input: { email: "tom@makeit-studio.com" }) { + message + } +} + +mutation signInWithCode { + signInEmailCodeConfirm(input: { email: "tom@makeit-studio.com", code: 731560 }) { + idToken + } +} + +query me { + me { + _id + } +} + +mutation completeRegistration { + completeRegistration(input: { firstName: "Tom", lastName: "Roelants", organisation: "Makeit", position: "Dev" }) { + _id + } +} diff --git a/__tests/admin/config.ts b/__tests/admin/config.ts new file mode 100644 index 0000000..a329877 --- /dev/null +++ b/__tests/admin/config.ts @@ -0,0 +1,36 @@ +import { Thunder } from './helpers/graphql-zeus'; +import fetch from 'node-fetch'; + +const url = 'http://localhost:4001'; +const token = + 'eyJhbGciOiJSUzI1NiIsImtpZCI6IjVmOTcxMmEwODczMTcyMGQ2NmZkNGEyYTU5MmU0ZGZjMmI1ZGU1OTUiLCJ0eXAiOiJKV1QifQ.eyJpc3MiOiJodHRwczovL3NlY3VyZXRva2VuLmdvb2dsZS5jb20vc29mdHdhcmUtdGVhbS1zdGFnaW5nIiwiYXVkIjoic29mdHdhcmUtdGVhbS1zdGFnaW5nIiwiYXV0aF90aW1lIjoxNjEwNzE1ODA4LCJ1c2VyX2lkIjoiTDNpQ3RkZDRzb2djYVJ1Y1RXbVVJNjM0aWo1MyIsInN1YiI6IkwzaUN0ZGQ0c29nY2FSdWNUV21VSTYzNGlqNTMiLCJpYXQiOjE2MTA3MTU4MDgsImV4cCI6MTYxMDcxOTQwOCwiZW1haWwiOiJzc3llZC5tYWtlaXRAZ21haWwuY29tIiwiZW1haWxfdmVyaWZpZWQiOnRydWUsImZpcmViYXNlIjp7ImlkZW50aXRpZXMiOnsiZW1haWwiOlsic3N5ZWQubWFrZWl0QGdtYWlsLmNvbSJdfSwic2lnbl9pbl9wcm92aWRlciI6ImN1c3RvbSJ9fQ.o3HXVp5lrIk_bzaTqb-aA7bpRJLb-Jp7lPz7_O7PQdsOaw5-L066HFtp5I-meS2CdpvUXy1REvAntGEeTNFukBzOkcRjBYagaxwUd_KpZOOoru5wpvTW8jJXlzmQXmDbHm2zfmzSCxsmrdZz62p_mKKJgGw5QTZB_LxUqTo6v0rhboXOowTzvAaHf1iwSU84JfrZwKU4xd0031RZPFbi6NDgHprYIDx7L5Fsl6bQhXDEXELDe4RNYQ5b0CKXa8rdJE9rBuroGJAJhB42_iwDwCa07FwBLIJzXefKqI97s15s23RzVT5o81EtIDOlOI6nBhucsad4boxacT6VBuS_vw'; + +export const thunder = Thunder(async (query) => { + const headers: any = { + 'Content-Type': 'application/json', + }; + + if (token) headers.authorization = token; + + const response = await fetch(url, { + body: JSON.stringify({ query }), + method: 'POST', + headers: headers, + }); + if (!response.ok) { + return new Promise((resolve, reject) => { + response + .text() + .then((text) => { + try { + reject(JSON.parse(text)); + } catch (err) { + reject(text); + } + }) + .catch(reject); + }); + } + const json = await response.json(); + return json.data; +}); diff --git a/__tests/admin/helpers/graphql-zeus.ts b/__tests/admin/helpers/graphql-zeus.ts new file mode 100644 index 0000000..925a88a --- /dev/null +++ b/__tests/admin/helpers/graphql-zeus.ts @@ -0,0 +1,4164 @@ +/* tslint:disable */ +/* eslint-disable */ + +export type ValueTypes = { + ["AccountGenderEnum"]:AccountGenderEnum; + ["AccountModel"]: AliasType<{ + _id?:true, + organisationId?:true, + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + createdAt?:true, + updatedAt?:true, + r?:true, + w?:true, + d?:true, + types?:true, + organisationIds?:true, + email?:true, + userName?:true, + firstName?:true, + lastName?:true, + profilePicture?:ValueTypes["ImageComponent"], + phoneNumber?:true, + gender?:true, + birthDate?:true, + getAge?:true, + address?:ValueTypes["AddressComponent"], + language?:true, + settings?:ValueTypes["SettingsComponent"], + organisation?:true, + position?:true, + profileCompleted?:true, + __typename?: true +}>; + ["AccountTypeEnum"]:AccountTypeEnum; + ["AddressComponent"]: AliasType<{ + number?:true, + street?:true, + streetBis?:true, + floor?:true, + box?:true, + zip?:true, + state?:true, + city?:true, + country?:true, + __typename?: true +}>; + ["AddressInput"]: { + number?:string, + street?:string, + streetBis?:string, + floor?:string, + box?:string, + zip?:string, + state?:string, + city:string, + country:string +}; + ["AvailableTranslation"]:AvailableTranslation; + ["CategoryModel"]: AliasType<{ + _id?:true, + organisationId?:true, + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + createdAt?:true, + updatedAt?:true, + r?:true, + w?:true, + d?:true, + title?:ValueTypes["TranslatableComponent"], + teaser?:ValueTypes["TranslatableComponent"], + cover?:ValueTypes["ImageComponent"], + thumbnail?:ValueTypes["ImageComponent"], + extraImages?:ValueTypes["ImageComponent"], + content?:ValueTypes["TranslatableComponent"], + seo?:ValueTypes["SEOField"], + urls?:ValueTypes["TranslatableComponent"], + ressourceType?:true, + colorCode?:true, + __typename?: true +}>; + ["CompleteAccountInput"]: { + firstName:string, + lastName:string, + organisation:string, + position:string +}; + /** The javascript `Date` as string. Type represents date and time as the ISO Date string. */ +["DateTime"]:unknown; + ["EditAccountInput"]: { + userName?:string, + firstName?:string, + lastName?:string, + profilePicture?:ValueTypes["ImageInput"], + phoneNumber?:string, + gender?:ValueTypes["AccountGenderEnum"], + birthDate?:ValueTypes["DateTime"], + address?:ValueTypes["AddressInput"], + language?:ValueTypes["AvailableTranslation"], + settings?:ValueTypes["SettingsInput"][], + organisation?:string, + position?:string +}; + ["EditCategoryInput"]: { + ressourceType:ValueTypes["RessourceEnum"], + title:ValueTypes["TranslatableInput"], + content?:ValueTypes["TranslatableInput"], + colorCode?:string +}; + ["EditEmailSettingsInput"]: { + subject?:ValueTypes["TranslatableInput"], + body?:ValueTypes["TranslatableInput"], + type?:string, + custom?:boolean, + templateId?:ValueTypes["TranslatableInput"], + fromEmail:string, + fromName:string, + replyToEmail:string, + cci?:string[], + smsSettings?:ValueTypes["SmsSettingsComponentInput"], + slackSettings?:ValueTypes["SlackSettingsComponentInput"] +}; + ["EditListInput"]: { + title?:ValueTypes["TranslatableInput"], + teaser?:ValueTypes["TranslatableInput"], + cover?:ValueTypes["ImageInput"], + thumbnail?:ValueTypes["ImageInput"], + content?:ValueTypes["TranslatableInput"], + seo?:ValueTypes["SEOInput"], + urls?:ValueTypes["TranslatableInput"], + ressourceType?:ValueTypes["ListEnum"], + colorCode?:string +}; + ["EditPageInput"]: { + title?:string, + label?:string, + content?:ValueTypes["JSONObject"], + position?:number, + toComplete?:boolean +}; + ["EditProjectSettingsInput"]: { + value?:string +}; + ["EmailSettingsModel"]: AliasType<{ + _id?:true, + organisationId?:true, + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + createdAt?:true, + updatedAt?:true, + r?:true, + w?:true, + d?:true, + type?:true, + key?:true, + subject?:ValueTypes["TranslatableComponent"], + body?:ValueTypes["TranslatableComponent"], + custom?:true, + templateId?:ValueTypes["TranslatableComponent"], + fromEmail?:true, + fromName?:true, + replyToEmail?:true, + cci?:true, + availableFields?:true, + templateSample?:true, + smsSettings?:ValueTypes["SmsSettingsComponent"], + pushSettings?:ValueTypes["PushNotifComponent"], + slackSettings?:ValueTypes["SlackSettingsComponent"], + __typename?: true +}>; + ["EnginePathComponent"]: AliasType<{ + ressourceModel?:true, + ressourceId?:true, + __typename?: true +}>; + ["FirebaseTokenResult"]: AliasType<{ + localId?:true, + email?:true, + displayName?:true, + idToken?:true, + registered?:true, + refreshToken?:true, + expiresIn?:true, + __typename?: true +}>; + ["GetArgs"]: { + limit:number, + skip:number, + sort?:string +}; + ["IEngineSchema"]:AliasType<{ + _id?:true, + organisationId?:true, + paths?:ValueTypes["EnginePathComponent"], + by?:ValueTypes["MetaBy"], + permissions?:ValueTypes["MetaPermissions"], + createdAt?:true, + updatedAt?:true; + ['...on NaceSchema']?: Omit; + ['...on PageSchema']?: Omit; + __typename?: true +}>; + ["ImageComponent"]: AliasType<{ + title?:true, + fileType?:true, + large?:true, + medium?:true, + small?:true, + __typename?: true +}>; + ["ImageInput"]: { + title?:string, + fileType?:string, + large:string, + medium?:string, + small?:string +}; + /** The `JSONObject` scalar type represents JSON objects as specified by [ECMA-404](http://www.ecma-international.org/publications/files/ECMA-ST/ECMA-404.pdf). */ +["JSONObject"]:unknown; + ["LevelEnum"]:LevelEnum; + ["LinkEmailInput"]: { + email:string, + password:string, + idToken:string +}; + ["ListEnum"]:ListEnum; + ["ListModel"]: AliasType<{ + _id?:true, + organisationId?:true, + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + createdAt?:true, + updatedAt?:true, + r?:true, + w?:true, + d?:true, + title?:ValueTypes["TranslatableComponent"], + teaser?:ValueTypes["TranslatableComponent"], + cover?:ValueTypes["ImageComponent"], + thumbnail?:ValueTypes["ImageComponent"], + extraImages?:ValueTypes["ImageComponent"], + content?:ValueTypes["TranslatableComponent"], + seo?:ValueTypes["SEOField"], + urls?:ValueTypes["TranslatableComponent"], + ressourceType?:true, + colorCode?:true, + __typename?: true +}>; + ["LoginInput"]: { + email:string, + password:string +}; + ["MetaBy"]: AliasType<{ + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + __typename?: true +}>; + ["MetaPermissions"]: AliasType<{ + r?:true, + w?:true, + d?:true, + __typename?: true +}>; + ["ModelLoadersEnum"]:ModelLoadersEnum; + ["Mutation"]: AliasType<{ +updateMe?: [{ input:ValueTypes["EditAccountInput"]},ValueTypes["AccountModel"]], +updateMeEmail?: [{ input:ValueTypes["NewEmailInput"]},ValueTypes["AccountModel"]], +updateMePassword?: [{ input:ValueTypes["NewPasswordInput"]},ValueTypes["AccountModel"]], +resetPassword?: [{ input:ValueTypes["ResetPasswordInput"]},ValueTypes["SimpleResult"]], +registerGuest?: [{ otherInfo:ValueTypes["EditAccountInput"], input:ValueTypes["LinkEmailInput"]},ValueTypes["AccountModel"]], +accountsAddOne?: [{ input:ValueTypes["NewAccountInputAdmin"]},ValueTypes["AccountModel"]], +accountsEditOne?: [{ input:ValueTypes["EditAccountInput"], id:string},ValueTypes["AccountModel"]], +accountsDeleteOne?: [{ userType:ValueTypes["AccountTypeEnum"], id:string},ValueTypes["AccountModel"]], +completeRegistration?: [{ input:ValueTypes["CompleteAccountInput"]},ValueTypes["AccountModel"]], +categoriesAddOne?: [{ input:ValueTypes["NewCategoryInput"]},ValueTypes["CategoryModel"]], +categoriesEditOne?: [{ input:ValueTypes["EditCategoryInput"], id:string},ValueTypes["CategoryModel"]], +categoriesDeleteOne?: [{ id:string},ValueTypes["CategoryModel"]], +emailsSettingsAddOne?: [{ input:ValueTypes["NewEmailSettingsInput"]},ValueTypes["EmailSettingsModel"]], +emailsSettingsEditOne?: [{ input:ValueTypes["EditEmailSettingsInput"], id:string},ValueTypes["EmailSettingsModel"]], +emailsSettingsDeleteOne?: [{ id:string},ValueTypes["EmailSettingsModel"]], +projectSettingsAddOne?: [{ input:ValueTypes["NewProjectSettingsInput"]},ValueTypes["ProjectSettingModel"]], +projectSettingsEditOne?: [{ input:ValueTypes["EditProjectSettingsInput"], id:string},ValueTypes["ProjectSettingModel"]], +projectSettingsDeleteOne?: [{ id:string},ValueTypes["ProjectSettingModel"]], +listsAddOne?: [{ input:ValueTypes["NewListInput"]},ValueTypes["ListModel"]], +listsEditOne?: [{ input:ValueTypes["EditListInput"], id:string},ValueTypes["ListModel"]], +listsDeleteOne?: [{ id:string},ValueTypes["ListModel"]], +pagesAddOne?: [{ input:ValueTypes["NewPageInput"]},ValueTypes["PageSchema"]], +pagesEditOne?: [{ input:ValueTypes["EditPageInput"], id:string},ValueTypes["PageSchema"]], +pagesDeleteOne?: [{ id:string},ValueTypes["PageSchema"]], + refreshCaching?:ValueTypes["SuccessResponse"], +naceAddOne?: [{ input:ValueTypes["NaceNewInputSchema"]},ValueTypes["NaceSchema"]], +naceEditOne?: [{ input:ValueTypes["NaceEditInputSchema"], id:string},ValueTypes["NaceSchema"]], +naceDeleteOne?: [{ id:string},ValueTypes["NaceSchema"]], + __typename?: true +}>; + ["NaceEditInputSchema"]: { + name:ValueTypes["TranslatableInput"], + code:string, + macroSelector:string, + description:ValueTypes["TranslatableInput"], + level:ValueTypes["LevelEnum"] +}; + ["NaceNewInputSchema"]: { + name:ValueTypes["TranslatableInput"], + code:string, + macroSelector:string, + description:ValueTypes["TranslatableInput"], + level:ValueTypes["LevelEnum"] +}; + ["NaceSchema"]: AliasType<{ + name?:ValueTypes["TranslatableComponent"], + code?:true, + macroSelector?:true, + description?:ValueTypes["TranslatableComponent"], + level?:true, + _id?:true, + organisationId?:true, + paths?:ValueTypes["EnginePathComponent"], + by?:ValueTypes["MetaBy"], + permissions?:ValueTypes["MetaPermissions"], + createdAt?:true, + updatedAt?:true, + __typename?: true +}>; + ["NewAccountInputAdmin"]: { + type:ValueTypes["AccountTypeEnum"], + organisationIds?:string[], + email:string, + userName?:string, + firstName?:string, + lastName?:string, + profilePicture?:ValueTypes["ImageInput"], + phoneNumber?:string, + gender?:ValueTypes["AccountGenderEnum"], + birthDate?:ValueTypes["DateTime"], + address?:ValueTypes["AddressInput"], + language?:ValueTypes["AvailableTranslation"], + settings?:ValueTypes["SettingsInput"][] +}; + ["NewCategoryInput"]: { + ressourceType:ValueTypes["RessourceEnum"], + title:ValueTypes["TranslatableInput"], + content?:ValueTypes["TranslatableInput"], + colorCode?:string +}; + ["NewEmailInput"]: { + newEmail:string +}; + ["NewEmailSettingsInput"]: { + subject?:ValueTypes["TranslatableInput"], + body?:ValueTypes["TranslatableInput"], + type?:string, + key:string, + custom?:boolean, + templateId?:ValueTypes["TranslatableInput"], + fromEmail:string, + fromName:string, + replyToEmail:string, + cci?:string[], + smsSettings?:ValueTypes["SmsSettingsComponentInput"], + slackSettings?:ValueTypes["SlackSettingsComponentInput"] +}; + ["NewListInput"]: { + title:ValueTypes["TranslatableInput"], + teaser?:ValueTypes["TranslatableInput"], + cover?:ValueTypes["ImageInput"], + thumbnail?:ValueTypes["ImageInput"], + content?:ValueTypes["TranslatableInput"], + seo?:ValueTypes["SEOInput"], + urls?:ValueTypes["TranslatableInput"], + ressourceType:ValueTypes["ListEnum"], + colorCode?:string +}; + ["NewPageInput"]: { + title:string, + label?:string, + content?:ValueTypes["JSONObject"], + position?:number, + toComplete?:boolean +}; + ["NewPasswordInput"]: { + oldPassword:string, + newPassword:string, + newPasswordConfirmation:string +}; + ["NewProjectSettingsInput"]: { + key:string, + value:string +}; + ["PageSchema"]: AliasType<{ + title?:true, + label?:true, + content?:true, + position?:true, + toComplete?:true, + _id?:true, + organisationId?:true, + paths?:ValueTypes["EnginePathComponent"], + by?:ValueTypes["MetaBy"], + permissions?:ValueTypes["MetaPermissions"], + createdAt?:true, + updatedAt?:true, + __typename?: true +}>; + ["ProjectSettingModel"]: AliasType<{ + _id?:true, + organisationId?:true, + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + createdAt?:true, + updatedAt?:true, + r?:true, + w?:true, + d?:true, + type?:true, + key?:true, + value?:true, + __typename?: true +}>; + ["PushNotifComponent"]: AliasType<{ + subject?:ValueTypes["TranslatableComponent"], + text?:ValueTypes["TranslatableComponent"], + __typename?: true +}>; + ["Query"]: AliasType<{ + me?:ValueTypes["AccountModel"], +login?: [{ refreshToken?:string, creds?:ValueTypes["LoginInput"]},ValueTypes["FirebaseTokenResult"]], +magicLink?: [{ input:ValueTypes["ResetPasswordInput"]},ValueTypes["SimpleResult"]], +accountsGetOne?: [{ id:string},ValueTypes["AccountModel"]], +accountsGetMany?: [{ pagination?:ValueTypes["GetArgs"]},ValueTypes["AccountModel"]], +categoriesGetOne?: [{ id:string},ValueTypes["CategoryModel"]], +categoriesGetMany?: [{ search?:string, ressourceType:ValueTypes["RessourceEnum"], pagination?:ValueTypes["GetArgs"]},ValueTypes["CategoryModel"]], +categoriesGetCount?: [{ search?:string, ressourceType:ValueTypes["RessourceEnum"], pagination?:ValueTypes["GetArgs"]},true], +emailsSettingsGetOne?: [{ id:string},ValueTypes["EmailSettingsModel"]], +emailsSettingsGetMany?: [{ search?:string, pagination?:ValueTypes["GetArgs"]},ValueTypes["EmailSettingsModel"]], +emailsSettingsGetCount?: [{ search?:string, pagination?:ValueTypes["GetArgs"]},true], +projectSettingsGetOne?: [{ id:string},ValueTypes["ProjectSettingModel"]], +projectSettingsGetMany?: [{ _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"], type?:ValueTypes["SettingsType"]},ValueTypes["ProjectSettingModel"]], +projectSettingsGetCount?: [{ _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"], type?:ValueTypes["SettingsType"]},true], +listsGetOne?: [{ id:string},ValueTypes["ListModel"]], +listsGetMany?: [{ search?:string, ressourceType:ValueTypes["ListEnum"], pagination?:ValueTypes["GetArgs"]},ValueTypes["ListModel"]], +listsGetCount?: [{ search?:string, ressourceType:ValueTypes["ListEnum"], pagination?:ValueTypes["GetArgs"]},true], +pagesGetOne?: [{ id:string},ValueTypes["PageSchema"]], +pagesGetMany?: [{ _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"]},ValueTypes["PageSchema"]], +pagesGetCount?: [{ _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"]},true], +naceGetOne?: [{ id:string},ValueTypes["NaceSchema"]], +naceGetMany?: [{ _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"], name?:ValueTypes["TranslatableInput"], code?:string, macroSelector?:string, level?:ValueTypes["LevelEnum"]},ValueTypes["NaceSchema"]], +naceGetCount?: [{ _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"], name?:ValueTypes["TranslatableInput"], code?:string, macroSelector?:string, level?:ValueTypes["LevelEnum"]},true], + __typename?: true +}>; + ["ResetPasswordInput"]: { + email:string +}; + ["RessourceEnum"]:RessourceEnum; + ["SEOField"]: AliasType<{ + title?:ValueTypes["TranslatableComponent"], + description?:ValueTypes["TranslatableComponent"], + keywords?:ValueTypes["TranslatableComponent"], + thumbnail?:ValueTypes["ImageComponent"], + __typename?: true +}>; + ["SEOInput"]: { + title:ValueTypes["TranslatableInput"], + description:ValueTypes["TranslatableInput"], + keywords?:ValueTypes["TranslatableInput"], + thumbnail?:ValueTypes["ImageInput"] +}; + ["SettingsComponent"]: AliasType<{ + type?:true, + email?:true, + pushNotifications?:true, + sms?:true, + __typename?: true +}>; + ["SettingsInput"]: { + type:ValueTypes["SettingsTypeEnum"], + email?:boolean, + pushNotifications?:boolean, + sms?:boolean +}; + ["SettingsType"]:SettingsType; + ["SettingsTypeEnum"]:SettingsTypeEnum; + ["SimpleResult"]: AliasType<{ + message?:true, + __typename?: true +}>; + ["SlackActionInfo"]: AliasType<{ + text?:true, + url?:true, + style?:true, + __typename?: true +}>; + ["SlackActionInfoInput"]: { + text:string, + url:string, + style?:ValueTypes["SlackActionStyleEnum"] +}; + ["SlackActionStyleEnum"]:SlackActionStyleEnum; + ["SlackSettingsComponent"]: AliasType<{ + webhookUrls?:true, + username?:true, + text?:true, + icon?:true, + actions?:ValueTypes["SlackActionInfo"], + __typename?: true +}>; + ["SlackSettingsComponentInput"]: { + webhookUrls:string[], + username?:string, + text:string, + icon?:string, + actions?:ValueTypes["SlackActionInfoInput"][] +}; + ["SmsSettingsComponent"]: AliasType<{ + text?:ValueTypes["TranslatableComponent"], + __typename?: true +}>; + ["SmsSettingsComponentInput"]: { + text:ValueTypes["TranslatableInput"] +}; + ["SuccessResponse"]: AliasType<{ + success?:true, + __typename?: true +}>; + ["TranslatableComponent"]: AliasType<{ + en?:true, + __typename?: true +}>; + ["TranslatableInput"]: { + en?:string +} + } + +export type PartialObjects = { + ["AccountGenderEnum"]:AccountGenderEnum, + ["AccountModel"]: { + __typename?: "AccountModel"; + _id?:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt?:PartialObjects["DateTime"], + updatedAt?:PartialObjects["DateTime"], + r?:string[], + w?:string[], + d?:string[], + types?:PartialObjects["AccountTypeEnum"][], + organisationIds?:string[], + email?:string, + userName?:string, + firstName?:string, + lastName?:string, + profilePicture?:PartialObjects["ImageComponent"], + phoneNumber?:string, + gender?:PartialObjects["AccountGenderEnum"], + birthDate?:PartialObjects["DateTime"], + getAge?:number, + address?:PartialObjects["AddressComponent"], + language?:PartialObjects["AvailableTranslation"], + settings?:PartialObjects["SettingsComponent"][], + organisation?:string, + position?:string, + profileCompleted?:boolean + }, + ["AccountTypeEnum"]:AccountTypeEnum, + ["AddressComponent"]: { + __typename?: "AddressComponent"; + number?:string, + street?:string, + streetBis?:string, + floor?:string, + box?:string, + zip?:string, + state?:string, + city?:string, + country?:string + }, + ["AddressInput"]: { + number?:string, + street?:string, + streetBis?:string, + floor?:string, + box?:string, + zip?:string, + state?:string, + city:string, + country:string +}, + ["AvailableTranslation"]:AvailableTranslation, + ["CategoryModel"]: { + __typename?: "CategoryModel"; + _id?:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt?:PartialObjects["DateTime"], + updatedAt?:PartialObjects["DateTime"], + r?:string[], + w?:string[], + d?:string[], + title?:PartialObjects["TranslatableComponent"], + teaser?:PartialObjects["TranslatableComponent"], + cover?:PartialObjects["ImageComponent"], + thumbnail?:PartialObjects["ImageComponent"], + extraImages?:PartialObjects["ImageComponent"][], + content?:PartialObjects["TranslatableComponent"], + seo?:PartialObjects["SEOField"], + urls?:PartialObjects["TranslatableComponent"], + ressourceType?:PartialObjects["RessourceEnum"], + colorCode?:string + }, + ["CompleteAccountInput"]: { + firstName:string, + lastName:string, + organisation:string, + position:string +}, + /** The javascript `Date` as string. Type represents date and time as the ISO Date string. */ +["DateTime"]:any, + ["EditAccountInput"]: { + userName?:string, + firstName?:string, + lastName?:string, + profilePicture?:PartialObjects["ImageInput"], + phoneNumber?:string, + gender?:PartialObjects["AccountGenderEnum"], + birthDate?:PartialObjects["DateTime"], + address?:PartialObjects["AddressInput"], + language?:PartialObjects["AvailableTranslation"], + settings?:PartialObjects["SettingsInput"][], + organisation?:string, + position?:string +}, + ["EditCategoryInput"]: { + ressourceType:PartialObjects["RessourceEnum"], + title:PartialObjects["TranslatableInput"], + content?:PartialObjects["TranslatableInput"], + colorCode?:string +}, + ["EditEmailSettingsInput"]: { + subject?:PartialObjects["TranslatableInput"], + body?:PartialObjects["TranslatableInput"], + type?:string, + custom?:boolean, + templateId?:PartialObjects["TranslatableInput"], + fromEmail:string, + fromName:string, + replyToEmail:string, + cci?:string[], + smsSettings?:PartialObjects["SmsSettingsComponentInput"], + slackSettings?:PartialObjects["SlackSettingsComponentInput"] +}, + ["EditListInput"]: { + title?:PartialObjects["TranslatableInput"], + teaser?:PartialObjects["TranslatableInput"], + cover?:PartialObjects["ImageInput"], + thumbnail?:PartialObjects["ImageInput"], + content?:PartialObjects["TranslatableInput"], + seo?:PartialObjects["SEOInput"], + urls?:PartialObjects["TranslatableInput"], + ressourceType?:PartialObjects["ListEnum"], + colorCode?:string +}, + ["EditPageInput"]: { + title?:string, + label?:string, + content?:PartialObjects["JSONObject"], + position?:number, + toComplete?:boolean +}, + ["EditProjectSettingsInput"]: { + value?:string +}, + ["EmailSettingsModel"]: { + __typename?: "EmailSettingsModel"; + _id?:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt?:PartialObjects["DateTime"], + updatedAt?:PartialObjects["DateTime"], + r?:string[], + w?:string[], + d?:string[], + type?:PartialObjects["SettingsType"], + key?:string, + subject?:PartialObjects["TranslatableComponent"], + body?:PartialObjects["TranslatableComponent"], + custom?:boolean, + templateId?:PartialObjects["TranslatableComponent"], + fromEmail?:string, + fromName?:string, + replyToEmail?:string, + cci?:string[], + availableFields?:string[], + templateSample?:string, + smsSettings?:PartialObjects["SmsSettingsComponent"], + pushSettings?:PartialObjects["PushNotifComponent"], + slackSettings?:PartialObjects["SlackSettingsComponent"] + }, + ["EnginePathComponent"]: { + __typename?: "EnginePathComponent"; + ressourceModel?:PartialObjects["ModelLoadersEnum"], + ressourceId?:string + }, + ["FirebaseTokenResult"]: { + __typename?: "FirebaseTokenResult"; + localId?:string, + email?:string, + displayName?:string, + idToken?:string, + registered?:boolean, + refreshToken?:string, + expiresIn?:string + }, + ["GetArgs"]: { + limit:number, + skip:number, + sort?:string +}, + ["IEngineSchema"]:{ + _id?:string; + organisationId?:string; + paths?:PartialObjects["EnginePathComponent"][]; + by?:PartialObjects["MetaBy"]; + permissions?:PartialObjects["MetaPermissions"]; + createdAt?:PartialObjects["DateTime"]; + updatedAt?:PartialObjects["DateTime"] +} & (PartialObjects["NaceSchema"] | PartialObjects["PageSchema"]), + ["ImageComponent"]: { + __typename?: "ImageComponent"; + title?:string, + fileType?:string, + large?:string, + medium?:string, + small?:string + }, + ["ImageInput"]: { + title?:string, + fileType?:string, + large:string, + medium?:string, + small?:string +}, + /** The `JSONObject` scalar type represents JSON objects as specified by [ECMA-404](http://www.ecma-international.org/publications/files/ECMA-ST/ECMA-404.pdf). */ +["JSONObject"]:any, + ["LevelEnum"]:LevelEnum, + ["LinkEmailInput"]: { + email:string, + password:string, + idToken:string +}, + ["ListEnum"]:ListEnum, + ["ListModel"]: { + __typename?: "ListModel"; + _id?:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt?:PartialObjects["DateTime"], + updatedAt?:PartialObjects["DateTime"], + r?:string[], + w?:string[], + d?:string[], + title?:PartialObjects["TranslatableComponent"], + teaser?:PartialObjects["TranslatableComponent"], + cover?:PartialObjects["ImageComponent"], + thumbnail?:PartialObjects["ImageComponent"], + extraImages?:PartialObjects["ImageComponent"][], + content?:PartialObjects["TranslatableComponent"], + seo?:PartialObjects["SEOField"], + urls?:PartialObjects["TranslatableComponent"], + ressourceType?:PartialObjects["ListEnum"], + colorCode?:string + }, + ["LoginInput"]: { + email:string, + password:string +}, + ["MetaBy"]: { + __typename?: "MetaBy"; + createdBy?:string, + updatedBy?:string, + deletedBy?:string + }, + ["MetaPermissions"]: { + __typename?: "MetaPermissions"; + r?:string[], + w?:string[], + d?:string[] + }, + ["ModelLoadersEnum"]:ModelLoadersEnum, + ["Mutation"]: { + __typename?: "Mutation"; + updateMe?:PartialObjects["AccountModel"], + updateMeEmail?:PartialObjects["AccountModel"], + updateMePassword?:PartialObjects["AccountModel"], + resetPassword?:PartialObjects["SimpleResult"], + registerGuest?:PartialObjects["AccountModel"], + accountsAddOne?:PartialObjects["AccountModel"], + accountsEditOne?:PartialObjects["AccountModel"], + accountsDeleteOne?:PartialObjects["AccountModel"], + completeRegistration?:PartialObjects["AccountModel"], + categoriesAddOne?:PartialObjects["CategoryModel"], + categoriesEditOne?:PartialObjects["CategoryModel"], + categoriesDeleteOne?:PartialObjects["CategoryModel"], + emailsSettingsAddOne?:PartialObjects["EmailSettingsModel"], + emailsSettingsEditOne?:PartialObjects["EmailSettingsModel"], + emailsSettingsDeleteOne?:PartialObjects["EmailSettingsModel"], + projectSettingsAddOne?:PartialObjects["ProjectSettingModel"], + projectSettingsEditOne?:PartialObjects["ProjectSettingModel"], + projectSettingsDeleteOne?:PartialObjects["ProjectSettingModel"], + listsAddOne?:PartialObjects["ListModel"], + listsEditOne?:PartialObjects["ListModel"], + listsDeleteOne?:PartialObjects["ListModel"], + pagesAddOne?:PartialObjects["PageSchema"], + pagesEditOne?:PartialObjects["PageSchema"], + pagesDeleteOne?:PartialObjects["PageSchema"], + refreshCaching?:PartialObjects["SuccessResponse"], + naceAddOne?:PartialObjects["NaceSchema"], + naceEditOne?:PartialObjects["NaceSchema"], + naceDeleteOne?:PartialObjects["NaceSchema"] + }, + ["NaceEditInputSchema"]: { + name:PartialObjects["TranslatableInput"], + code:string, + macroSelector:string, + description:PartialObjects["TranslatableInput"], + level:PartialObjects["LevelEnum"] +}, + ["NaceNewInputSchema"]: { + name:PartialObjects["TranslatableInput"], + code:string, + macroSelector:string, + description:PartialObjects["TranslatableInput"], + level:PartialObjects["LevelEnum"] +}, + ["NaceSchema"]: { + __typename?: "NaceSchema"; + name?:PartialObjects["TranslatableComponent"], + code?:string, + macroSelector?:string, + description?:PartialObjects["TranslatableComponent"], + level?:PartialObjects["LevelEnum"], + _id?:string, + organisationId?:string, + paths?:PartialObjects["EnginePathComponent"][], + by?:PartialObjects["MetaBy"], + permissions?:PartialObjects["MetaPermissions"], + createdAt?:PartialObjects["DateTime"], + updatedAt?:PartialObjects["DateTime"] + }, + ["NewAccountInputAdmin"]: { + type:PartialObjects["AccountTypeEnum"], + organisationIds?:string[], + email:string, + userName?:string, + firstName?:string, + lastName?:string, + profilePicture?:PartialObjects["ImageInput"], + phoneNumber?:string, + gender?:PartialObjects["AccountGenderEnum"], + birthDate?:PartialObjects["DateTime"], + address?:PartialObjects["AddressInput"], + language?:PartialObjects["AvailableTranslation"], + settings?:PartialObjects["SettingsInput"][] +}, + ["NewCategoryInput"]: { + ressourceType:PartialObjects["RessourceEnum"], + title:PartialObjects["TranslatableInput"], + content?:PartialObjects["TranslatableInput"], + colorCode?:string +}, + ["NewEmailInput"]: { + newEmail:string +}, + ["NewEmailSettingsInput"]: { + subject?:PartialObjects["TranslatableInput"], + body?:PartialObjects["TranslatableInput"], + type?:string, + key:string, + custom?:boolean, + templateId?:PartialObjects["TranslatableInput"], + fromEmail:string, + fromName:string, + replyToEmail:string, + cci?:string[], + smsSettings?:PartialObjects["SmsSettingsComponentInput"], + slackSettings?:PartialObjects["SlackSettingsComponentInput"] +}, + ["NewListInput"]: { + title:PartialObjects["TranslatableInput"], + teaser?:PartialObjects["TranslatableInput"], + cover?:PartialObjects["ImageInput"], + thumbnail?:PartialObjects["ImageInput"], + content?:PartialObjects["TranslatableInput"], + seo?:PartialObjects["SEOInput"], + urls?:PartialObjects["TranslatableInput"], + ressourceType:PartialObjects["ListEnum"], + colorCode?:string +}, + ["NewPageInput"]: { + title:string, + label?:string, + content?:PartialObjects["JSONObject"], + position?:number, + toComplete?:boolean +}, + ["NewPasswordInput"]: { + oldPassword:string, + newPassword:string, + newPasswordConfirmation:string +}, + ["NewProjectSettingsInput"]: { + key:string, + value:string +}, + ["PageSchema"]: { + __typename?: "PageSchema"; + title?:string, + label?:string, + content?:PartialObjects["JSONObject"], + position?:number, + toComplete?:boolean, + _id?:string, + organisationId?:string, + paths?:PartialObjects["EnginePathComponent"][], + by?:PartialObjects["MetaBy"], + permissions?:PartialObjects["MetaPermissions"], + createdAt?:PartialObjects["DateTime"], + updatedAt?:PartialObjects["DateTime"] + }, + ["ProjectSettingModel"]: { + __typename?: "ProjectSettingModel"; + _id?:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt?:PartialObjects["DateTime"], + updatedAt?:PartialObjects["DateTime"], + r?:string[], + w?:string[], + d?:string[], + type?:PartialObjects["SettingsType"], + key?:string, + value?:string + }, + ["PushNotifComponent"]: { + __typename?: "PushNotifComponent"; + subject?:PartialObjects["TranslatableComponent"], + text?:PartialObjects["TranslatableComponent"] + }, + ["Query"]: { + __typename?: "Query"; + me?:PartialObjects["AccountModel"], + login?:PartialObjects["FirebaseTokenResult"], + magicLink?:PartialObjects["SimpleResult"], + accountsGetOne?:PartialObjects["AccountModel"], + accountsGetMany?:PartialObjects["AccountModel"][], + categoriesGetOne?:PartialObjects["CategoryModel"], + categoriesGetMany?:PartialObjects["CategoryModel"][], + categoriesGetCount?:number, + emailsSettingsGetOne?:PartialObjects["EmailSettingsModel"], + emailsSettingsGetMany?:PartialObjects["EmailSettingsModel"][], + emailsSettingsGetCount?:number, + projectSettingsGetOne?:PartialObjects["ProjectSettingModel"], + projectSettingsGetMany?:PartialObjects["ProjectSettingModel"][], + projectSettingsGetCount?:number, + listsGetOne?:PartialObjects["ListModel"], + listsGetMany?:PartialObjects["ListModel"][], + listsGetCount?:number, + pagesGetOne?:PartialObjects["PageSchema"], + pagesGetMany?:PartialObjects["PageSchema"][], + pagesGetCount?:number, + naceGetOne?:PartialObjects["NaceSchema"], + naceGetMany?:PartialObjects["NaceSchema"][], + naceGetCount?:number + }, + ["ResetPasswordInput"]: { + email:string +}, + ["RessourceEnum"]:RessourceEnum, + ["SEOField"]: { + __typename?: "SEOField"; + title?:PartialObjects["TranslatableComponent"], + description?:PartialObjects["TranslatableComponent"], + keywords?:PartialObjects["TranslatableComponent"], + thumbnail?:PartialObjects["ImageComponent"] + }, + ["SEOInput"]: { + title:PartialObjects["TranslatableInput"], + description:PartialObjects["TranslatableInput"], + keywords?:PartialObjects["TranslatableInput"], + thumbnail?:PartialObjects["ImageInput"] +}, + ["SettingsComponent"]: { + __typename?: "SettingsComponent"; + type?:PartialObjects["SettingsTypeEnum"], + email?:boolean, + pushNotifications?:boolean, + sms?:boolean + }, + ["SettingsInput"]: { + type:PartialObjects["SettingsTypeEnum"], + email?:boolean, + pushNotifications?:boolean, + sms?:boolean +}, + ["SettingsType"]:SettingsType, + ["SettingsTypeEnum"]:SettingsTypeEnum, + ["SimpleResult"]: { + __typename?: "SimpleResult"; + message?:string + }, + ["SlackActionInfo"]: { + __typename?: "SlackActionInfo"; + text?:string, + url?:string, + style?:PartialObjects["SlackActionStyleEnum"] + }, + ["SlackActionInfoInput"]: { + text:string, + url:string, + style?:PartialObjects["SlackActionStyleEnum"] +}, + ["SlackActionStyleEnum"]:SlackActionStyleEnum, + ["SlackSettingsComponent"]: { + __typename?: "SlackSettingsComponent"; + webhookUrls?:string[], + username?:string, + text?:string, + icon?:string, + actions?:PartialObjects["SlackActionInfo"][] + }, + ["SlackSettingsComponentInput"]: { + webhookUrls:string[], + username?:string, + text:string, + icon?:string, + actions?:PartialObjects["SlackActionInfoInput"][] +}, + ["SmsSettingsComponent"]: { + __typename?: "SmsSettingsComponent"; + text?:PartialObjects["TranslatableComponent"] + }, + ["SmsSettingsComponentInput"]: { + text:PartialObjects["TranslatableInput"] +}, + ["SuccessResponse"]: { + __typename?: "SuccessResponse"; + success?:boolean + }, + ["TranslatableComponent"]: { + __typename?: "TranslatableComponent"; + en?:string + }, + ["TranslatableInput"]: { + en?:string +} + } + +export enum AccountGenderEnum { + m = "m", + f = "f", + other = "other" +} + +export type AccountModel = { + __typename?: "AccountModel", + _id:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt:DateTime, + updatedAt:DateTime, + r:string[], + w:string[], + d:string[], + types:AccountTypeEnum[], + organisationIds?:string[], + email:string, + userName?:string, + firstName:string, + lastName:string, + profilePicture?:ImageComponent, + phoneNumber?:string, + gender?:AccountGenderEnum, + birthDate?:DateTime, + getAge?:number, + address?:AddressComponent, + language?:AvailableTranslation, + settings?:SettingsComponent[], + organisation:string, + position:string, + profileCompleted:boolean +} + +export enum AccountTypeEnum { + admin = "admin", + user = "user", + public = "public", + blogAuthor = "blogAuthor", + investor = "investor", + company = "company", + bank = "bank", + consultant = "consultant" +} + +export type AddressComponent = { + __typename?: "AddressComponent", + number?:string, + street?:string, + streetBis?:string, + floor?:string, + box?:string, + zip?:string, + state?:string, + city:string, + country:string +} + +export type AddressInput = { + number?:string, + street?:string, + streetBis?:string, + floor?:string, + box?:string, + zip?:string, + state?:string, + city:string, + country:string +} + +export enum AvailableTranslation { + en = "en" +} + +export type CategoryModel = { + __typename?: "CategoryModel", + _id:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt:DateTime, + updatedAt:DateTime, + r:string[], + w:string[], + d:string[], + title:TranslatableComponent, + teaser?:TranslatableComponent, + cover?:ImageComponent, + thumbnail?:ImageComponent, + extraImages?:ImageComponent[], + content?:TranslatableComponent, + seo?:SEOField, + urls?:TranslatableComponent, + ressourceType:RessourceEnum, + colorCode?:string +} + +export type CompleteAccountInput = { + firstName:string, + lastName:string, + organisation:string, + position:string +} + +/** The javascript `Date` as string. Type represents date and time as the ISO Date string. */ +export type DateTime = any + +export type EditAccountInput = { + userName?:string, + firstName?:string, + lastName?:string, + profilePicture?:ImageInput, + phoneNumber?:string, + gender?:AccountGenderEnum, + birthDate?:DateTime, + address?:AddressInput, + language?:AvailableTranslation, + settings?:SettingsInput[], + organisation?:string, + position?:string +} + +export type EditCategoryInput = { + ressourceType:RessourceEnum, + title:TranslatableInput, + content?:TranslatableInput, + colorCode?:string +} + +export type EditEmailSettingsInput = { + subject?:TranslatableInput, + body?:TranslatableInput, + type?:string, + custom?:boolean, + templateId?:TranslatableInput, + fromEmail:string, + fromName:string, + replyToEmail:string, + cci?:string[], + smsSettings?:SmsSettingsComponentInput, + slackSettings?:SlackSettingsComponentInput +} + +export type EditListInput = { + title?:TranslatableInput, + teaser?:TranslatableInput, + cover?:ImageInput, + thumbnail?:ImageInput, + content?:TranslatableInput, + seo?:SEOInput, + urls?:TranslatableInput, + ressourceType?:ListEnum, + colorCode?:string +} + +export type EditPageInput = { + title?:string, + label?:string, + content?:JSONObject, + position?:number, + toComplete?:boolean +} + +export type EditProjectSettingsInput = { + value?:string +} + +export type EmailSettingsModel = { + __typename?: "EmailSettingsModel", + _id:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt:DateTime, + updatedAt:DateTime, + r:string[], + w:string[], + d:string[], + type:SettingsType, + key:string, + subject?:TranslatableComponent, + body?:TranslatableComponent, + custom:boolean, + templateId?:TranslatableComponent, + fromEmail:string, + fromName:string, + replyToEmail:string, + cci?:string[], + availableFields?:string[], + templateSample?:string, + smsSettings?:SmsSettingsComponent, + pushSettings?:PushNotifComponent, + slackSettings?:SlackSettingsComponent +} + +export type EnginePathComponent = { + __typename?: "EnginePathComponent", + ressourceModel:ModelLoadersEnum, + ressourceId:string +} + +export type FirebaseTokenResult = { + __typename?: "FirebaseTokenResult", + localId:string, + email?:string, + displayName?:string, + idToken:string, + registered?:boolean, + refreshToken:string, + expiresIn:string +} + +export type GetArgs = { + limit:number, + skip:number, + sort?:string +} + +export type IEngineSchema = { + __interface:{ + _id:string, + organisationId?:string, + paths?:EnginePathComponent[], + by?:MetaBy, + permissions?:MetaPermissions, + createdAt:DateTime, + updatedAt:DateTime + }; + __resolve:{ + ['...on NaceSchema']: NaceSchema; + ['...on PageSchema']: PageSchema; + } +} + +export type ImageComponent = { + __typename?: "ImageComponent", + title?:string, + fileType?:string, + large:string, + medium?:string, + small?:string +} + +export type ImageInput = { + title?:string, + fileType?:string, + large:string, + medium?:string, + small?:string +} + +/** The `JSONObject` scalar type represents JSON objects as specified by [ECMA-404](http://www.ecma-international.org/publications/files/ECMA-ST/ECMA-404.pdf). */ +export type JSONObject = any + +export enum LevelEnum { + level1 = "level1", + level2 = "level2", + level3 = "level3", + level4 = "level4" +} + +export type LinkEmailInput = { + email:string, + password:string, + idToken:string +} + +export enum ListEnum { + articles = "articles" +} + +export type ListModel = { + __typename?: "ListModel", + _id:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt:DateTime, + updatedAt:DateTime, + r:string[], + w:string[], + d:string[], + title:TranslatableComponent, + teaser?:TranslatableComponent, + cover?:ImageComponent, + thumbnail?:ImageComponent, + extraImages?:ImageComponent[], + content?:TranslatableComponent, + seo?:SEOField, + urls?:TranslatableComponent, + ressourceType:ListEnum, + colorCode?:string +} + +export type LoginInput = { + email:string, + password:string +} + +export type MetaBy = { + __typename?: "MetaBy", + createdBy?:string, + updatedBy?:string, + deletedBy?:string +} + +export type MetaPermissions = { + __typename?: "MetaPermissions", + r:string[], + w:string[], + d:string[] +} + +export enum ModelLoadersEnum { + streams = "streams", + notifications = "notifications", + pages = "pages", + accounts = "accounts", + companies = "companies", + nace = "nace" +} + +export type Mutation = { + __typename?: "Mutation", + updateMe:AccountModel, + updateMeEmail:AccountModel, + updateMePassword:AccountModel, + resetPassword:SimpleResult, + registerGuest:AccountModel, + accountsAddOne:AccountModel, + accountsEditOne:AccountModel, + accountsDeleteOne:AccountModel, + completeRegistration:AccountModel, + categoriesAddOne:CategoryModel, + categoriesEditOne:CategoryModel, + categoriesDeleteOne:CategoryModel, + emailsSettingsAddOne:EmailSettingsModel, + emailsSettingsEditOne:EmailSettingsModel, + emailsSettingsDeleteOne:EmailSettingsModel, + projectSettingsAddOne:ProjectSettingModel, + projectSettingsEditOne:ProjectSettingModel, + projectSettingsDeleteOne:ProjectSettingModel, + listsAddOne:ListModel, + listsEditOne:ListModel, + listsDeleteOne:ListModel, + pagesAddOne:PageSchema, + pagesEditOne:PageSchema, + pagesDeleteOne:PageSchema, + refreshCaching:SuccessResponse, + naceAddOne:NaceSchema, + naceEditOne:NaceSchema, + naceDeleteOne:NaceSchema +} + +export type NaceEditInputSchema = { + name:TranslatableInput, + code:string, + macroSelector:string, + description:TranslatableInput, + level:LevelEnum +} + +export type NaceNewInputSchema = { + name:TranslatableInput, + code:string, + macroSelector:string, + description:TranslatableInput, + level:LevelEnum +} + +export type NaceSchema = { + __typename?: "NaceSchema", + name:TranslatableComponent, + code:string, + macroSelector:string, + description:TranslatableComponent, + level:LevelEnum, + _id:string, + organisationId?:string, + paths?:EnginePathComponent[], + by?:MetaBy, + permissions?:MetaPermissions, + createdAt:DateTime, + updatedAt:DateTime +} + +export type NewAccountInputAdmin = { + type:AccountTypeEnum, + organisationIds?:string[], + email:string, + userName?:string, + firstName?:string, + lastName?:string, + profilePicture?:ImageInput, + phoneNumber?:string, + gender?:AccountGenderEnum, + birthDate?:DateTime, + address?:AddressInput, + language?:AvailableTranslation, + settings?:SettingsInput[] +} + +export type NewCategoryInput = { + ressourceType:RessourceEnum, + title:TranslatableInput, + content?:TranslatableInput, + colorCode?:string +} + +export type NewEmailInput = { + newEmail:string +} + +export type NewEmailSettingsInput = { + subject?:TranslatableInput, + body?:TranslatableInput, + type?:string, + key:string, + custom?:boolean, + templateId?:TranslatableInput, + fromEmail:string, + fromName:string, + replyToEmail:string, + cci?:string[], + smsSettings?:SmsSettingsComponentInput, + slackSettings?:SlackSettingsComponentInput +} + +export type NewListInput = { + title:TranslatableInput, + teaser?:TranslatableInput, + cover?:ImageInput, + thumbnail?:ImageInput, + content?:TranslatableInput, + seo?:SEOInput, + urls?:TranslatableInput, + ressourceType:ListEnum, + colorCode?:string +} + +export type NewPageInput = { + title:string, + label?:string, + content?:JSONObject, + position?:number, + toComplete?:boolean +} + +export type NewPasswordInput = { + oldPassword:string, + newPassword:string, + newPasswordConfirmation:string +} + +export type NewProjectSettingsInput = { + key:string, + value:string +} + +export type PageSchema = { + __typename?: "PageSchema", + title:string, + label?:string, + content?:JSONObject, + position?:number, + toComplete?:boolean, + _id:string, + organisationId?:string, + paths?:EnginePathComponent[], + by?:MetaBy, + permissions?:MetaPermissions, + createdAt:DateTime, + updatedAt:DateTime +} + +export type ProjectSettingModel = { + __typename?: "ProjectSettingModel", + _id:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt:DateTime, + updatedAt:DateTime, + r:string[], + w:string[], + d:string[], + type:SettingsType, + key:string, + value:string +} + +export type PushNotifComponent = { + __typename?: "PushNotifComponent", + subject?:TranslatableComponent, + text:TranslatableComponent +} + +export type Query = { + __typename?: "Query", + me:AccountModel, + login:FirebaseTokenResult, + magicLink:SimpleResult, + accountsGetOne:AccountModel, + accountsGetMany:AccountModel[], + categoriesGetOne:CategoryModel, + categoriesGetMany:CategoryModel[], + categoriesGetCount:number, + emailsSettingsGetOne:EmailSettingsModel, + emailsSettingsGetMany:EmailSettingsModel[], + emailsSettingsGetCount:number, + projectSettingsGetOne:ProjectSettingModel, + projectSettingsGetMany:ProjectSettingModel[], + projectSettingsGetCount:number, + listsGetOne:ListModel, + listsGetMany:ListModel[], + listsGetCount:number, + pagesGetOne:PageSchema, + pagesGetMany:PageSchema[], + pagesGetCount:number, + naceGetOne:NaceSchema, + naceGetMany:NaceSchema[], + naceGetCount:number +} + +export type ResetPasswordInput = { + email:string +} + +export enum RessourceEnum { + none = "none" +} + +export type SEOField = { + __typename?: "SEOField", + title:TranslatableComponent, + description:TranslatableComponent, + keywords?:TranslatableComponent, + thumbnail?:ImageComponent +} + +export type SEOInput = { + title:TranslatableInput, + description:TranslatableInput, + keywords?:TranslatableInput, + thumbnail?:ImageInput +} + +export type SettingsComponent = { + __typename?: "SettingsComponent", + type:SettingsTypeEnum, + email?:boolean, + pushNotifications?:boolean, + sms?:boolean +} + +export type SettingsInput = { + type:SettingsTypeEnum, + email?:boolean, + pushNotifications?:boolean, + sms?:boolean +} + +export enum SettingsType { + emails = "emails", + ac = "ac", + env = "env", + instagram = "instagram", + others = "others" +} + +export enum SettingsTypeEnum { + notification = "notification" +} + +export type SimpleResult = { + __typename?: "SimpleResult", + message:string +} + +export type SlackActionInfo = { + __typename?: "SlackActionInfo", + text:string, + url:string, + style?:SlackActionStyleEnum +} + +export type SlackActionInfoInput = { + text:string, + url:string, + style?:SlackActionStyleEnum +} + +export enum SlackActionStyleEnum { + primary = "primary", + danger = "danger" +} + +export type SlackSettingsComponent = { + __typename?: "SlackSettingsComponent", + webhookUrls:string[], + username?:string, + text:string, + icon?:string, + actions?:SlackActionInfo[] +} + +export type SlackSettingsComponentInput = { + webhookUrls:string[], + username?:string, + text:string, + icon?:string, + actions?:SlackActionInfoInput[] +} + +export type SmsSettingsComponent = { + __typename?: "SmsSettingsComponent", + text:TranslatableComponent +} + +export type SmsSettingsComponentInput = { + text:TranslatableInput +} + +export type SuccessResponse = { + __typename?: "SuccessResponse", + success:boolean +} + +export type TranslatableComponent = { + __typename?: "TranslatableComponent", + en?:string +} + +export type TranslatableInput = { + en?:string +} + +export const AllTypesProps: Record = { + AccountGenderEnum: "enum", + AccountTypeEnum: "enum", + AddressInput:{ + number:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + street:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + streetBis:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + floor:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + box:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + zip:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + state:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + city:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + country:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + AvailableTranslation: "enum", + CompleteAccountInput:{ + firstName:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + lastName:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + organisation:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + position:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + DateTime: "String", + EditAccountInput:{ + userName:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + firstName:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + lastName:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + profilePicture:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + phoneNumber:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + gender:{ + type:"AccountGenderEnum", + array:false, + arrayRequired:false, + required:false + }, + birthDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + address:{ + type:"AddressInput", + array:false, + arrayRequired:false, + required:false + }, + language:{ + type:"AvailableTranslation", + array:false, + arrayRequired:false, + required:false + }, + settings:{ + type:"SettingsInput", + array:true, + arrayRequired:false, + required:true + }, + organisation:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + position:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + EditCategoryInput:{ + ressourceType:{ + type:"RessourceEnum", + array:false, + arrayRequired:false, + required:true + }, + title:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:true + }, + content:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:false + }, + colorCode:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + EditEmailSettingsInput:{ + subject:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:false + }, + body:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:false + }, + type:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + custom:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + }, + templateId:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:false + }, + fromEmail:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + fromName:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + replyToEmail:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + cci:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + smsSettings:{ + type:"SmsSettingsComponentInput", + array:false, + arrayRequired:false, + required:false + }, + slackSettings:{ + type:"SlackSettingsComponentInput", + array:false, + arrayRequired:false, + required:false + } + }, + EditListInput:{ + title:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:false + }, + teaser:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:false + }, + cover:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + thumbnail:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + content:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:false + }, + seo:{ + type:"SEOInput", + array:false, + arrayRequired:false, + required:false + }, + urls:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:false + }, + ressourceType:{ + type:"ListEnum", + array:false, + arrayRequired:false, + required:false + }, + colorCode:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + EditPageInput:{ + title:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + label:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + content:{ + type:"JSONObject", + array:false, + arrayRequired:false, + required:false + }, + position:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + toComplete:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + } + }, + EditProjectSettingsInput:{ + value:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + GetArgs:{ + limit:{ + type:"Int", + array:false, + arrayRequired:false, + required:true + }, + skip:{ + type:"Int", + array:false, + arrayRequired:false, + required:true + }, + sort:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + ImageInput:{ + title:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + fileType:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + large:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + medium:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + small:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + JSONObject: "String", + LevelEnum: "enum", + LinkEmailInput:{ + email:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + password:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + idToken:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + ListEnum: "enum", + LoginInput:{ + email:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + password:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + ModelLoadersEnum: "enum", + Mutation:{ + updateMe:{ + input:{ + type:"EditAccountInput", + array:false, + arrayRequired:false, + required:true + } + }, + updateMeEmail:{ + input:{ + type:"NewEmailInput", + array:false, + arrayRequired:false, + required:true + } + }, + updateMePassword:{ + input:{ + type:"NewPasswordInput", + array:false, + arrayRequired:false, + required:true + } + }, + resetPassword:{ + input:{ + type:"ResetPasswordInput", + array:false, + arrayRequired:false, + required:true + } + }, + registerGuest:{ + otherInfo:{ + type:"EditAccountInput", + array:false, + arrayRequired:false, + required:true + }, + input:{ + type:"LinkEmailInput", + array:false, + arrayRequired:false, + required:true + } + }, + accountsAddOne:{ + input:{ + type:"NewAccountInputAdmin", + array:false, + arrayRequired:false, + required:true + } + }, + accountsEditOne:{ + input:{ + type:"EditAccountInput", + array:false, + arrayRequired:false, + required:true + }, + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + accountsDeleteOne:{ + userType:{ + type:"AccountTypeEnum", + array:false, + arrayRequired:false, + required:true + }, + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + completeRegistration:{ + input:{ + type:"CompleteAccountInput", + array:false, + arrayRequired:false, + required:true + } + }, + categoriesAddOne:{ + input:{ + type:"NewCategoryInput", + array:false, + arrayRequired:false, + required:true + } + }, + categoriesEditOne:{ + input:{ + type:"EditCategoryInput", + array:false, + arrayRequired:false, + required:true + }, + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + categoriesDeleteOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + emailsSettingsAddOne:{ + input:{ + type:"NewEmailSettingsInput", + array:false, + arrayRequired:false, + required:true + } + }, + emailsSettingsEditOne:{ + input:{ + type:"EditEmailSettingsInput", + array:false, + arrayRequired:false, + required:true + }, + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + emailsSettingsDeleteOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + projectSettingsAddOne:{ + input:{ + type:"NewProjectSettingsInput", + array:false, + arrayRequired:false, + required:true + } + }, + projectSettingsEditOne:{ + input:{ + type:"EditProjectSettingsInput", + array:false, + arrayRequired:false, + required:true + }, + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + projectSettingsDeleteOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + listsAddOne:{ + input:{ + type:"NewListInput", + array:false, + arrayRequired:false, + required:true + } + }, + listsEditOne:{ + input:{ + type:"EditListInput", + array:false, + arrayRequired:false, + required:true + }, + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + listsDeleteOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + pagesAddOne:{ + input:{ + type:"NewPageInput", + array:false, + arrayRequired:false, + required:true + } + }, + pagesEditOne:{ + input:{ + type:"EditPageInput", + array:false, + arrayRequired:false, + required:true + }, + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + pagesDeleteOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + naceAddOne:{ + input:{ + type:"NaceNewInputSchema", + array:false, + arrayRequired:false, + required:true + } + }, + naceEditOne:{ + input:{ + type:"NaceEditInputSchema", + array:false, + arrayRequired:false, + required:true + }, + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + naceDeleteOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + } + }, + NaceEditInputSchema:{ + name:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:true + }, + code:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + macroSelector:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + description:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:true + }, + level:{ + type:"LevelEnum", + array:false, + arrayRequired:false, + required:true + } + }, + NaceNewInputSchema:{ + name:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:true + }, + code:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + macroSelector:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + description:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:true + }, + level:{ + type:"LevelEnum", + array:false, + arrayRequired:false, + required:true + } + }, + NewAccountInputAdmin:{ + type:{ + type:"AccountTypeEnum", + array:false, + arrayRequired:false, + required:true + }, + organisationIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + email:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + userName:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + firstName:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + lastName:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + profilePicture:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + phoneNumber:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + gender:{ + type:"AccountGenderEnum", + array:false, + arrayRequired:false, + required:false + }, + birthDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + address:{ + type:"AddressInput", + array:false, + arrayRequired:false, + required:false + }, + language:{ + type:"AvailableTranslation", + array:false, + arrayRequired:false, + required:false + }, + settings:{ + type:"SettingsInput", + array:true, + arrayRequired:false, + required:true + } + }, + NewCategoryInput:{ + ressourceType:{ + type:"RessourceEnum", + array:false, + arrayRequired:false, + required:true + }, + title:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:true + }, + content:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:false + }, + colorCode:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + NewEmailInput:{ + newEmail:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + NewEmailSettingsInput:{ + subject:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:false + }, + body:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:false + }, + type:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + key:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + custom:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + }, + templateId:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:false + }, + fromEmail:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + fromName:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + replyToEmail:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + cci:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + smsSettings:{ + type:"SmsSettingsComponentInput", + array:false, + arrayRequired:false, + required:false + }, + slackSettings:{ + type:"SlackSettingsComponentInput", + array:false, + arrayRequired:false, + required:false + } + }, + NewListInput:{ + title:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:true + }, + teaser:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:false + }, + cover:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + thumbnail:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + content:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:false + }, + seo:{ + type:"SEOInput", + array:false, + arrayRequired:false, + required:false + }, + urls:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:false + }, + ressourceType:{ + type:"ListEnum", + array:false, + arrayRequired:false, + required:true + }, + colorCode:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + NewPageInput:{ + title:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + label:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + content:{ + type:"JSONObject", + array:false, + arrayRequired:false, + required:false + }, + position:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + toComplete:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + } + }, + NewPasswordInput:{ + oldPassword:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + newPassword:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + newPasswordConfirmation:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + NewProjectSettingsInput:{ + key:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + value:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + Query:{ + login:{ + refreshToken:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + creds:{ + type:"LoginInput", + array:false, + arrayRequired:false, + required:false + } + }, + magicLink:{ + input:{ + type:"ResetPasswordInput", + array:false, + arrayRequired:false, + required:true + } + }, + accountsGetOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + accountsGetMany:{ + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + }, + categoriesGetOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + categoriesGetMany:{ + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + ressourceType:{ + type:"RessourceEnum", + array:false, + arrayRequired:false, + required:true + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + }, + categoriesGetCount:{ + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + ressourceType:{ + type:"RessourceEnum", + array:false, + arrayRequired:false, + required:true + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + }, + emailsSettingsGetOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + emailsSettingsGetMany:{ + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + }, + emailsSettingsGetCount:{ + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + }, + projectSettingsGetOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + projectSettingsGetMany:{ + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + type:{ + type:"SettingsType", + array:false, + arrayRequired:false, + required:false + } + }, + projectSettingsGetCount:{ + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + type:{ + type:"SettingsType", + array:false, + arrayRequired:false, + required:false + } + }, + listsGetOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + listsGetMany:{ + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + ressourceType:{ + type:"ListEnum", + array:false, + arrayRequired:false, + required:true + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + }, + listsGetCount:{ + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + ressourceType:{ + type:"ListEnum", + array:false, + arrayRequired:false, + required:true + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + }, + pagesGetOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + pagesGetMany:{ + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + }, + pagesGetCount:{ + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + }, + naceGetOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + naceGetMany:{ + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + name:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:false + }, + code:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + macroSelector:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + level:{ + type:"LevelEnum", + array:false, + arrayRequired:false, + required:false + } + }, + naceGetCount:{ + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + name:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:false + }, + code:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + macroSelector:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + level:{ + type:"LevelEnum", + array:false, + arrayRequired:false, + required:false + } + } + }, + ResetPasswordInput:{ + email:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + RessourceEnum: "enum", + SEOInput:{ + title:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:true + }, + description:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:true + }, + keywords:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:false + }, + thumbnail:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + } + }, + SettingsInput:{ + type:{ + type:"SettingsTypeEnum", + array:false, + arrayRequired:false, + required:true + }, + email:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + }, + pushNotifications:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + }, + sms:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + } + }, + SettingsType: "enum", + SettingsTypeEnum: "enum", + SlackActionInfoInput:{ + text:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + url:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + style:{ + type:"SlackActionStyleEnum", + array:false, + arrayRequired:false, + required:false + } + }, + SlackActionStyleEnum: "enum", + SlackSettingsComponentInput:{ + webhookUrls:{ + type:"String", + array:true, + arrayRequired:true, + required:true + }, + username:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + text:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + icon:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + actions:{ + type:"SlackActionInfoInput", + array:true, + arrayRequired:false, + required:true + } + }, + SmsSettingsComponentInput:{ + text:{ + type:"TranslatableInput", + array:false, + arrayRequired:false, + required:true + } + }, + TranslatableInput:{ + en:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + } +} + +export const ReturnTypes: Record = { + AccountModel:{ + _id:"ID", + organisationId:"ID", + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID", + createdAt:"DateTime", + updatedAt:"DateTime", + r:"String", + w:"String", + d:"String", + types:"AccountTypeEnum", + organisationIds:"String", + email:"String", + userName:"String", + firstName:"String", + lastName:"String", + profilePicture:"ImageComponent", + phoneNumber:"String", + gender:"AccountGenderEnum", + birthDate:"DateTime", + getAge:"Int", + address:"AddressComponent", + language:"AvailableTranslation", + settings:"SettingsComponent", + organisation:"String", + position:"String", + profileCompleted:"Boolean" + }, + AddressComponent:{ + number:"String", + street:"String", + streetBis:"String", + floor:"String", + box:"String", + zip:"String", + state:"String", + city:"String", + country:"String" + }, + CategoryModel:{ + _id:"ID", + organisationId:"ID", + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID", + createdAt:"DateTime", + updatedAt:"DateTime", + r:"String", + w:"String", + d:"String", + title:"TranslatableComponent", + teaser:"TranslatableComponent", + cover:"ImageComponent", + thumbnail:"ImageComponent", + extraImages:"ImageComponent", + content:"TranslatableComponent", + seo:"SEOField", + urls:"TranslatableComponent", + ressourceType:"RessourceEnum", + colorCode:"String" + }, + EmailSettingsModel:{ + _id:"ID", + organisationId:"ID", + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID", + createdAt:"DateTime", + updatedAt:"DateTime", + r:"String", + w:"String", + d:"String", + type:"SettingsType", + key:"String", + subject:"TranslatableComponent", + body:"TranslatableComponent", + custom:"Boolean", + templateId:"TranslatableComponent", + fromEmail:"String", + fromName:"String", + replyToEmail:"String", + cci:"String", + availableFields:"String", + templateSample:"String", + smsSettings:"SmsSettingsComponent", + pushSettings:"PushNotifComponent", + slackSettings:"SlackSettingsComponent" + }, + EnginePathComponent:{ + ressourceModel:"ModelLoadersEnum", + ressourceId:"String" + }, + FirebaseTokenResult:{ + localId:"String", + email:"String", + displayName:"String", + idToken:"String", + registered:"Boolean", + refreshToken:"String", + expiresIn:"String" + }, + IEngineSchema:{ + "...on NaceSchema": "NaceSchema", + "...on PageSchema": "PageSchema", + _id:"ID", + organisationId:"ID", + paths:"EnginePathComponent", + by:"MetaBy", + permissions:"MetaPermissions", + createdAt:"DateTime", + updatedAt:"DateTime" + }, + ImageComponent:{ + title:"String", + fileType:"String", + large:"String", + medium:"String", + small:"String" + }, + ListModel:{ + _id:"ID", + organisationId:"ID", + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID", + createdAt:"DateTime", + updatedAt:"DateTime", + r:"String", + w:"String", + d:"String", + title:"TranslatableComponent", + teaser:"TranslatableComponent", + cover:"ImageComponent", + thumbnail:"ImageComponent", + extraImages:"ImageComponent", + content:"TranslatableComponent", + seo:"SEOField", + urls:"TranslatableComponent", + ressourceType:"ListEnum", + colorCode:"String" + }, + MetaBy:{ + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID" + }, + MetaPermissions:{ + r:"String", + w:"String", + d:"String" + }, + Mutation:{ + updateMe:"AccountModel", + updateMeEmail:"AccountModel", + updateMePassword:"AccountModel", + resetPassword:"SimpleResult", + registerGuest:"AccountModel", + accountsAddOne:"AccountModel", + accountsEditOne:"AccountModel", + accountsDeleteOne:"AccountModel", + completeRegistration:"AccountModel", + categoriesAddOne:"CategoryModel", + categoriesEditOne:"CategoryModel", + categoriesDeleteOne:"CategoryModel", + emailsSettingsAddOne:"EmailSettingsModel", + emailsSettingsEditOne:"EmailSettingsModel", + emailsSettingsDeleteOne:"EmailSettingsModel", + projectSettingsAddOne:"ProjectSettingModel", + projectSettingsEditOne:"ProjectSettingModel", + projectSettingsDeleteOne:"ProjectSettingModel", + listsAddOne:"ListModel", + listsEditOne:"ListModel", + listsDeleteOne:"ListModel", + pagesAddOne:"PageSchema", + pagesEditOne:"PageSchema", + pagesDeleteOne:"PageSchema", + refreshCaching:"SuccessResponse", + naceAddOne:"NaceSchema", + naceEditOne:"NaceSchema", + naceDeleteOne:"NaceSchema" + }, + NaceSchema:{ + name:"TranslatableComponent", + code:"String", + macroSelector:"String", + description:"TranslatableComponent", + level:"LevelEnum", + _id:"ID", + organisationId:"ID", + paths:"EnginePathComponent", + by:"MetaBy", + permissions:"MetaPermissions", + createdAt:"DateTime", + updatedAt:"DateTime" + }, + PageSchema:{ + title:"String", + label:"String", + content:"JSONObject", + position:"Int", + toComplete:"Boolean", + _id:"ID", + organisationId:"ID", + paths:"EnginePathComponent", + by:"MetaBy", + permissions:"MetaPermissions", + createdAt:"DateTime", + updatedAt:"DateTime" + }, + ProjectSettingModel:{ + _id:"ID", + organisationId:"ID", + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID", + createdAt:"DateTime", + updatedAt:"DateTime", + r:"String", + w:"String", + d:"String", + type:"SettingsType", + key:"String", + value:"String" + }, + PushNotifComponent:{ + subject:"TranslatableComponent", + text:"TranslatableComponent" + }, + Query:{ + me:"AccountModel", + login:"FirebaseTokenResult", + magicLink:"SimpleResult", + accountsGetOne:"AccountModel", + accountsGetMany:"AccountModel", + categoriesGetOne:"CategoryModel", + categoriesGetMany:"CategoryModel", + categoriesGetCount:"Float", + emailsSettingsGetOne:"EmailSettingsModel", + emailsSettingsGetMany:"EmailSettingsModel", + emailsSettingsGetCount:"Float", + projectSettingsGetOne:"ProjectSettingModel", + projectSettingsGetMany:"ProjectSettingModel", + projectSettingsGetCount:"Float", + listsGetOne:"ListModel", + listsGetMany:"ListModel", + listsGetCount:"Float", + pagesGetOne:"PageSchema", + pagesGetMany:"PageSchema", + pagesGetCount:"Float", + naceGetOne:"NaceSchema", + naceGetMany:"NaceSchema", + naceGetCount:"Float" + }, + SEOField:{ + title:"TranslatableComponent", + description:"TranslatableComponent", + keywords:"TranslatableComponent", + thumbnail:"ImageComponent" + }, + SettingsComponent:{ + type:"SettingsTypeEnum", + email:"Boolean", + pushNotifications:"Boolean", + sms:"Boolean" + }, + SimpleResult:{ + message:"String" + }, + SlackActionInfo:{ + text:"String", + url:"String", + style:"SlackActionStyleEnum" + }, + SlackSettingsComponent:{ + webhookUrls:"String", + username:"String", + text:"String", + icon:"String", + actions:"SlackActionInfo" + }, + SmsSettingsComponent:{ + text:"TranslatableComponent" + }, + SuccessResponse:{ + success:"Boolean" + }, + TranslatableComponent:{ + en:"String" + } +} + +export class GraphQLError extends Error { + constructor(public response: GraphQLResponse) { + super(""); + console.error(response); + } + toString() { + return "GraphQL Response Error"; + } + } + + + +export type UnwrapPromise = T extends Promise ? R : T; +export type ZeusState Promise> = NonNullable< + UnwrapPromise> +>; +export type ZeusHook< + T extends ( + ...args: any[] + ) => Record Promise>, + N extends keyof ReturnType +> = ZeusState[N]>; + +type Func

= (...args: P) => R; +type AnyFunc = Func; + +type WithTypeNameValue = T & { + __typename?: true; +}; + +type AliasType = WithTypeNameValue & { + __alias?: Record>; +}; + +type NotUndefined = T extends undefined ? never : T; + +export type ResolverType = NotUndefined; + +export type ArgsType = F extends Func ? P : never; + +interface GraphQLResponse { + data?: Record; + errors?: Array<{ + message: string; + }>; +} + +export type ValuesOf = T[keyof T]; + +export type MapResolve = SRC extends { + __interface: infer INTERFACE; + __resolve: Record & infer IMPLEMENTORS; + } + ? + ValuesOf<{ + [k in (keyof SRC['__resolve'] & keyof DST)]: ({ + [rk in (keyof SRC['__resolve'][k] & keyof DST[k])]: LastMapTypeSRCResolver + } & { + __typename: SRC['__resolve'][k]['__typename'] + }) + }> + : + never; + +export type MapInterface = SRC extends { + __interface: infer INTERFACE; + __resolve: Record & infer IMPLEMENTORS; + } + ? + (MapResolve extends never ? {} : MapResolve) & { + [k in (keyof SRC['__interface'] & keyof DST)]: LastMapTypeSRCResolver +} : never; + +export type ValueToUnion = T extends { + __typename: infer R; +} + ? { + [P in keyof Omit]: T[P] & { + __typename: R; + }; + } + : T; + +export type ObjectToUnion = { + [P in keyof T]: T[P]; +}[keyof T]; + +type Anify = { [P in keyof T]?: any }; + + +type LastMapTypeSRCResolver = SRC extends undefined + ? undefined + : SRC extends Array + ? LastMapTypeSRCResolver[] + : SRC extends { __interface: any; __resolve: any } + ? MapInterface + : SRC extends { __union: any; __resolve: infer RESOLVE } + ? ObjectToUnion>> + : DST extends boolean + ? SRC + : MapType; + +export type MapType, DST> = DST extends boolean + ? SRC + : DST extends { + __alias: any; + } + ? { + [A in keyof DST["__alias"]]: Required extends Anify< + DST["__alias"][A] + > + ? MapType, DST["__alias"][A]> + : never; + } & + { + [Key in keyof Omit]: DST[Key] extends [ + any, + infer PAYLOAD + ] + ? LastMapTypeSRCResolver + : LastMapTypeSRCResolver; + } + : { + [Key in keyof DST]: DST[Key] extends [any, infer PAYLOAD] + ? LastMapTypeSRCResolver + : LastMapTypeSRCResolver; + }; + +type OperationToGraphQL = (o: Z | V, variables?: Record) => Promise>; + +type CastToGraphQL = ( + resultOfYourQuery: any +) => (o: Z | V) => MapType; + +type fetchOptions = ArgsType; + +export type SelectionFunction = (t: T | V) => T; +type FetchFunction = (query: string, variables?: Record) => Promise; + + + +export const ZeusSelect = () => ((t: any) => t) as SelectionFunction; + +export const ScalarResolver = (scalar: string, value: any) => { + switch (scalar) { + case 'String': + return `${JSON.stringify(value)}`; + case 'Int': + return `${value}`; + case 'Float': + return `${value}`; + case 'Boolean': + return `${value}`; + case 'ID': + return `"${value}"`; + case 'enum': + return `${value}`; + case 'scalar': + return `${value}`; + default: + return false; + } +}; + + +export const TypesPropsResolver = ({ + value, + type, + name, + key, + blockArrays +}: { + value: any; + type: string; + name: string; + key?: string; + blockArrays?: boolean; +}): string => { + if (value === null) { + return `null`; + } + let resolvedValue = AllTypesProps[type][name]; + if (key) { + resolvedValue = resolvedValue[key]; + } + if (!resolvedValue) { + throw new Error(`Cannot resolve ${type} ${name}${key ? ` ${key}` : ''}`) + } + const typeResolved = resolvedValue.type; + const isArray = resolvedValue.array; + const isArrayRequired = resolvedValue.arrayRequired; + if (typeof value === 'string' && value.startsWith(`ZEUS_VAR$`)) { + const isRequired = resolvedValue.required ? '!' : ''; + let t = `${typeResolved}`; + if (isArray) { + if (isRequired) { + t = `${t}!`; + } + t = `[${t}]`; + if(isArrayRequired){ + t = `${t}!`; + } + }else{ + if (isRequired) { + t = `${t}!`; + } + } + return `\$${value.split(`ZEUS_VAR$`)[1]}__ZEUS_VAR__${t}`; + } + if (isArray && !blockArrays) { + return `[${value + .map((v: any) => TypesPropsResolver({ value: v, type, name, key, blockArrays: true })) + .join(',')}]`; + } + const reslovedScalar = ScalarResolver(typeResolved, value); + if (!reslovedScalar) { + const resolvedType = AllTypesProps[typeResolved]; + if (typeof resolvedType === 'object') { + const argsKeys = Object.keys(resolvedType); + return `{${argsKeys + .filter((ak) => value[ak] !== undefined) + .map( + (ak) => `${ak}:${TypesPropsResolver({ value: value[ak], type: typeResolved, name: ak })}` + )}}`; + } + return ScalarResolver(AllTypesProps[typeResolved], value) as string; + } + return reslovedScalar; +}; + + +const isArrayFunction = ( + parent: string[], + a: any[] +) => { + const [values, r] = a; + const [mainKey, key, ...keys] = parent; + const keyValues = Object.keys(values).filter((k) => typeof values[k] !== 'undefined'); + + if (!keys.length) { + return keyValues.length > 0 + ? `(${keyValues + .map( + (v) => + `${v}:${TypesPropsResolver({ + value: values[v], + type: mainKey, + name: key, + key: v + })}` + ) + .join(',')})${r ? traverseToSeekArrays(parent, r) : ''}` + : traverseToSeekArrays(parent, r); + } + + const [typeResolverKey] = keys.splice(keys.length - 1, 1); + let valueToResolve = ReturnTypes[mainKey][key]; + for (const k of keys) { + valueToResolve = ReturnTypes[valueToResolve][k]; + } + + const argumentString = + keyValues.length > 0 + ? `(${keyValues + .map( + (v) => + `${v}:${TypesPropsResolver({ + value: values[v], + type: valueToResolve, + name: typeResolverKey, + key: v + })}` + ) + .join(',')})${r ? traverseToSeekArrays(parent, r) : ''}` + : traverseToSeekArrays(parent, r); + return argumentString; +}; + + +const resolveKV = (k: string, v: boolean | string | { [x: string]: boolean | string }) => + typeof v === 'boolean' ? k : typeof v === 'object' ? `${k}{${objectToTree(v)}}` : `${k}${v}`; + + +const objectToTree = (o: { [x: string]: boolean | string }): string => + `{${Object.keys(o).map((k) => `${resolveKV(k, o[k])}`).join(' ')}}`; + + +const traverseToSeekArrays = (parent: string[], a?: any): string => { + if (!a) return ''; + if (Object.keys(a).length === 0) { + return ''; + } + let b: Record = {}; + if (Array.isArray(a)) { + return isArrayFunction([...parent], a); + } else { + if (typeof a === 'object') { + Object.keys(a) + .filter((k) => typeof a[k] !== 'undefined') + .map((k) => { + if (k === '__alias') { + Object.keys(a[k]).map((aliasKey) => { + const aliasOperations = a[k][aliasKey]; + const aliasOperationName = Object.keys(aliasOperations)[0]; + const aliasOperation = aliasOperations[aliasOperationName]; + b[ + `${aliasOperationName}__alias__${aliasKey}: ${aliasOperationName}` + ] = traverseToSeekArrays([...parent, aliasOperationName], aliasOperation); + }); + } else { + b[k] = traverseToSeekArrays([...parent, k], a[k]); + } + }); + } else { + return ''; + } + } + return objectToTree(b); +}; + + +const buildQuery = (type: string, a?: Record) => + traverseToSeekArrays([type], a); + + +const inspectVariables = (query: string) => { + const regex = /\$\b\w*__ZEUS_VAR__\[?[^!^\]^\s^,^\)^\}]*[!]?[\]]?[!]?/g; + let result; + const AllVariables: string[] = []; + while ((result = regex.exec(query))) { + if (AllVariables.includes(result[0])) { + continue; + } + AllVariables.push(result[0]); + } + if (!AllVariables.length) { + return query; + } + let filteredQuery = query; + AllVariables.forEach((variable) => { + while (filteredQuery.includes(variable)) { + filteredQuery = filteredQuery.replace(variable, variable.split('__ZEUS_VAR__')[0]); + } + }); + return `(${AllVariables.map((a) => a.split('__ZEUS_VAR__')) + .map(([variableName, variableType]) => `${variableName}:${variableType}`) + .join(', ')})${filteredQuery}`; +}; + + +const queryConstruct = (t: 'query' | 'mutation' | 'subscription', tName: string) => (o: Record) => + `${t.toLowerCase()}${inspectVariables(buildQuery(tName, o))}`; + + +const fullChainConstruct = (fn: FetchFunction) => (t: 'query' | 'mutation' | 'subscription', tName: string) => ( + o: Record, + variables?: Record, +) => fn(queryConstruct(t, tName)(o), variables).then((r:any) => { + seekForAliases(r) + return r +}); + + +const seekForAliases = (response: any) => { + const traverseAlias = (value: any) => { + if (Array.isArray(value)) { + value.forEach(seekForAliases); + } else { + if (typeof value === 'object') { + seekForAliases(value); + } + } + }; + if (typeof response === 'object' && response) { + const keys = Object.keys(response); + if (keys.length < 1) { + return; + } + keys.forEach((k) => { + const value = response[k]; + if (k.indexOf('__alias__') !== -1) { + const [operation, alias] = k.split('__alias__'); + response[alias] = { + [operation]: value, + }; + delete response[k]; + } + traverseAlias(value); + }); + } +}; + + +export const $ = (t: TemplateStringsArray): any => `ZEUS_VAR$${t.join('')}`; + + +const handleFetchResponse = ( + response: Parameters['then']>[0], Function>>[0] +): Promise => { + if (!response.ok) { + return new Promise((_, reject) => { + response.text().then(text => { + try { reject(JSON.parse(text)); } + catch (err) { reject(text); } + }).catch(reject); + }); + } + return response.json(); +}; + +const apiFetch = (options: fetchOptions) => (query: string, variables: Record = {}) => { + let fetchFunction = fetch; + let queryString = query; + let fetchOptions = options[1] || {}; + if (fetchOptions.method && fetchOptions.method === 'GET') { + queryString = encodeURIComponent(query); + return fetchFunction(`${options[0]}?query=${queryString}`, fetchOptions) + .then(handleFetchResponse) + .then((response: GraphQLResponse) => { + if (response.errors) { + throw new GraphQLError(response); + } + return response.data; + }); + } + return fetchFunction(`${options[0]}`, { + body: JSON.stringify({ query: queryString, variables }), + method: 'POST', + headers: { + 'Content-Type': 'application/json' + }, + ...fetchOptions + }) + .then(handleFetchResponse) + .then((response: GraphQLResponse) => { + if (response.errors) { + throw new GraphQLError(response); + } + return response.data; + }); + }; + + + +export const Thunder = (fn: FetchFunction) => ({ + query: ((o: any, variables) => + fullChainConstruct(fn)('query', 'Query')(o, variables).then( + (response: any) => response + )) as OperationToGraphQL, +mutation: ((o: any, variables) => + fullChainConstruct(fn)('mutation', 'Mutation')(o, variables).then( + (response: any) => response + )) as OperationToGraphQL +}); + +export const Chain = (...options: fetchOptions) => ({ + query: ((o: any, variables) => + fullChainConstruct(apiFetch(options))('query', 'Query')(o, variables).then( + (response: any) => response + )) as OperationToGraphQL, +mutation: ((o: any, variables) => + fullChainConstruct(apiFetch(options))('mutation', 'Mutation')(o, variables).then( + (response: any) => response + )) as OperationToGraphQL +}); +export const Zeus = { + query: (o:ValueTypes["Query"]) => queryConstruct('query', 'Query')(o), +mutation: (o:ValueTypes["Mutation"]) => queryConstruct('mutation', 'Mutation')(o) +}; +export const Cast = { + query: ((o: any) => (_: any) => o) as CastToGraphQL< + ValueTypes["Query"], + Query +>, +mutation: ((o: any) => (_: any) => o) as CastToGraphQL< + ValueTypes["Mutation"], + Mutation +> +}; +export const Selectors = { + query: ZeusSelect(), +mutation: ZeusSelect() +}; + + +export const Gql = Chain('http://localhost:4001') \ No newline at end of file diff --git a/__tests/admin/index.ts b/__tests/admin/index.ts new file mode 100644 index 0000000..e69de29 diff --git a/__tests/app/config.ts b/__tests/app/config.ts new file mode 100644 index 0000000..54d0d02 --- /dev/null +++ b/__tests/app/config.ts @@ -0,0 +1,7 @@ +import { Thunder } from './helpers/graphql-zeus'; +import { testRunQuery } from '../../lib/seed/__tests/helper'; + +const url = 'http://localhost:4000'; +const token = 'eyJhbGciOiJSUzI1NiIsImtpZCI6IjRlMDBlOGZlNWYyYzg4Y2YwYzcwNDRmMzA3ZjdlNzM5Nzg4ZTRmMWUiLCJ0eXAiOiJKV1QifQ.eyJpc3MiOiJodHRwczovL3NlY3VyZXRva2VuLmdvb2dsZS5jb20vd29ya2luZ2ZsZXgtcHJvZHVjdGlvbiIsImF1ZCI6IndvcmtpbmdmbGV4LXByb2R1Y3Rpb24iLCJhdXRoX3RpbWUiOjE2MTYxNzE2MTcsInVzZXJfaWQiOiI4TW1EdmZHbWV0Ylc5aVE5S0Iybm55TFRsV3gyIiwic3ViIjoiOE1tRHZmR21ldGJXOWlROUtCMm5ueUxUbFd4MiIsImlhdCI6MTYxNjE3MTYxNywiZXhwIjoxNjE2MTc1MjE3LCJlbWFpbCI6Im5pY29sYXMuYS5iZXJuaWVyK3dvcmtAZ21haWwuY29tIiwiZW1haWxfdmVyaWZpZWQiOnRydWUsImZpcmViYXNlIjp7ImlkZW50aXRpZXMiOnsiZW1haWwiOlsibmljb2xhcy5hLmJlcm5pZXIrd29ya0BnbWFpbC5jb20iXX0sInNpZ25faW5fcHJvdmlkZXIiOiJjdXN0b20ifX0.JQvWIyuMmh2tWtf6I1ACWadhyC7LbwvBdaqE0wtXy5YX2dj_cdrKyV1H35nmJ4xRXAoTfMIdarnfPqZaCcCDrAMVey4Jt2BU9dQ0COoQ-qz5KGzeFbeUU-MDSx3sP7sEm_qsdcgPvBHPAj2WnBjrUh58Je1JRVORcXbn9oopJ0v-U2sSn1H43Ycv5qDGF9eqPHaz3ex0NmVum77JRPIKGOKPopCDKrJgMG1tlUiSsr2Qo0YgmxuzY5SrIvVImXmyclNMNrEiRsemTACBScX9seDl_vKosbgIPFaTFqtJksI8NeKxBQFCKz78DwZpZPHxwlN7IEP-ilzIbbppFoHFIQ' + +export const thunder = Thunder(testRunQuery(url, token)); diff --git a/__tests/app/helpers/graphql-zeus.ts b/__tests/app/helpers/graphql-zeus.ts new file mode 100644 index 0000000..a046ef4 --- /dev/null +++ b/__tests/app/helpers/graphql-zeus.ts @@ -0,0 +1,6824 @@ +/* tslint:disable */ +/* eslint-disable */ + +export type ValueTypes = { + ["AccountGenderEnum"]:AccountGenderEnum; + ["AccountModel"]: AliasType<{ + _id?:true, + organisationId?:true, + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + createdAt?:true, + updatedAt?:true, + r?:true, + w?:true, + d?:true, + types?:true, + organisationIds?:true, + email?:true, + userName?:true, + firstName?:true, + lastName?:true, + profilePicture?:ValueTypes["ImageComponent"], + phoneNumber?:true, + gender?:true, + birthDate?:true, + getAge?:true, + address?:ValueTypes["AddressComponent"], + language?:true, + settings?:ValueTypes["SettingsComponent"], + paymentInfo?:ValueTypes["AccountPaymentInfo"], + payoutInfo?:ValueTypes["AccountPayoutInfo"], + terms?:true, + favLikes?:ValueTypes["FavoriteComponent"], + getFavorites?:ValueTypes["FavoriteComponent"], + getLiked?:ValueTypes["FavoriteComponent"], + __typename?: true +}>; + ["AccountPaymentInfo"]: AliasType<{ + stripeInfo?:ValueTypes["StripeInfo"], + paymentMethods?:ValueTypes["PaymentMethodDetail"], + billingInfos?:ValueTypes["BillingAddressComponent"], + __typename?: true +}>; + ["AccountPayoutInfo"]: AliasType<{ + stripeInfo?:ValueTypes["StripePayoutInfo"], + __typename?: true +}>; + ["AccountTypeEnum"]:AccountTypeEnum; + ["AddInput"]: { + ressourceId:string, + ressourceType:ValueTypes["RessourceEnum"] +}; + ["AddOnComponent"]: AliasType<{ + addOnId?:true, + quantity?:true, + __typename?: true +}>; + ["AddPeopleToRoomInput"]: { + roomId:string, + accountIds:string[] +}; + ["AddressComponent"]: AliasType<{ + number?:true, + street?:true, + streetBis?:true, + floor?:true, + box?:true, + zip?:true, + state?:true, + city?:true, + country?:true, + __typename?: true +}>; + ["AddressInput"]: { + number?:string, + street?:string, + streetBis?:string, + floor?:string, + box?:string, + zip?:string, + state?:string, + city:string, + country:string +}; + ["AddressStrictComponent"]: AliasType<{ + number?:true, + street?:true, + zip?:true, + city?:true, + country?:true, + streetBis?:true, + floor?:true, + box?:true, + state?:true, + __typename?: true +}>; + ["AddressStrictInput"]: { + number:string, + street:string, + zip:string, + city:string, + country:string, + streetBis?:string, + floor?:string, + box?:string, + state?:string +}; + ["AvailabilityComponent"]: AliasType<{ + dates?:ValueTypes["DateRangeComponent"], + hours?:ValueTypes["HourComponent"], + exceptions?:ValueTypes["DateExceptionComponent"], + noWeekend?:true, + __typename?: true +}>; + ["AvailabilityInput"]: { + dates:ValueTypes["DateRangeComponentInput"], + hours?:ValueTypes["HoursInput"][], + exceptions?:ValueTypes["DateExceptionComponentInput"][], + noWeekend?:boolean +}; + ["AvailableCurrency"]:AvailableCurrency; + ["AvailableTranslation"]:AvailableTranslation; + ["BillingAddressComponent"]: AliasType<{ + firstName?:true, + lastName?:true, + address?:ValueTypes["AddressStrictComponent"], + email?:true, + default?:true, + customTags?:true, + company?:true, + vatNumber?:true, + fiscalForm?:true, + _id?:true, + __typename?: true +}>; + ["BillingAddressInput"]: { + firstName:string, + lastName:string, + address:ValueTypes["AddressStrictInput"], + email?:string, + default?:boolean, + customTags?:ValueTypes["JSONObject"], + company?:string, + vatNumber?:string, + fiscalForm?:string +}; + ["BookingEngineSchema"]: AliasType<{ + durationType?:true, + startToEnd?:ValueTypes["DateRangeComponent"], + slot?:ValueTypes["SlotComponent"], + capacity?:true, + comments?:true, + status?:true, + paths?:ValueTypes["EnginePathComponent"], + ownerId?:true, + dates?:ValueTypes["DateRangeComponent"], + owner?:ValueTypes["AccountModel"], + _id?:true, + organisationId?:true, + by?:ValueTypes["MetaBy"], + permissions?:ValueTypes["MetaPermissions"], + createdAt?:true, + updatedAt?:true, + workspace?:ValueTypes["WorkspaceModel"], + room?:ValueTypes["RoomModel"], + __typename?: true +}>; + ["BookingNewInput"]: AliasType<{ + durationType?:true, + startToEnd?:ValueTypes["DateRangeComponent"], + slot?:ValueTypes["SlotComponent"], + capacity?:true, + comments?:true, + ressourceModel?:true, + ressourceId?:true, + checkoutInfo?:ValueTypes["CheckoutEngineInput"], + __typename?: true +}>; + ["BookingNewInputSchema"]: { + durationType:ValueTypes["DurationTypeEnum"], + startToEnd?:ValueTypes["DateRangeComponentInput"], + slot?:ValueTypes["SlotComponentInput"], + capacity?:number, + comments?:string, + ressourceModel:ValueTypes["BookingsRessourceEnum"], + ressourceId:string, + checkoutInfo?:ValueTypes["CheckoutEngineInputI"] +}; + ["BookingSchema"]: AliasType<{ + durationType?:true, + startToEnd?:ValueTypes["DateRangeComponent"], + slot?:ValueTypes["SlotComponent"], + capacity?:true, + comments?:true, + status?:true, + paths?:ValueTypes["EnginePathComponent"], + ownerId?:true, + dates?:ValueTypes["DateRangeComponent"], + _id?:true, + organisationId?:true, + by?:ValueTypes["MetaBy"], + permissions?:ValueTypes["MetaPermissions"], + createdAt?:true, + updatedAt?:true, + __typename?: true +}>; + ["BookingsRessourceEnum"]:BookingsRessourceEnum; + ["BuyerInfo"]: { + firstName:string, + lastName:string, + email:string, + company?:string, + vatNumber?:string, + address:ValueTypes["AddressInput"] +}; + ["BuyerInfoComponent"]: AliasType<{ + firstName?:true, + lastName?:true, + email?:true, + company?:true, + vatNumber?:true, + address?:ValueTypes["AddressComponent"], + __typename?: true +}>; + ["cardBrandEnum"]:cardBrandEnum; + ["CardInfo"]: AliasType<{ + brand?:true, + country?:true, + exp_month?:true, + exp_year?:true, + last4?:true, + __typename?: true +}>; + ["cardTypeEnum"]:cardTypeEnum; + ["CategoryModel"]: AliasType<{ + _id?:true, + organisationId?:true, + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + createdAt?:true, + updatedAt?:true, + r?:true, + w?:true, + d?:true, + title?:ValueTypes["TranslatableComponent"], + teaser?:ValueTypes["TranslatableComponent"], + cover?:ValueTypes["ImageComponent"], + thumbnail?:ValueTypes["ImageComponent"], + extraImages?:ValueTypes["ImageComponent"], + content?:ValueTypes["TranslatableComponent"], + seo?:ValueTypes["SEOField"], + urls?:ValueTypes["TranslatableComponent"], + ressourceType?:true, + colorCode?:true, + __typename?: true +}>; + ["CheckoutEngineInput"]: AliasType<{ + contactInfo?:ValueTypes["BuyerInfoComponent"], + billingInfo?:ValueTypes["BuyerInfoComponent"], + token?:true, + __typename?: true +}>; + ["CheckoutEngineInputI"]: { + contactInfo:ValueTypes["BuyerInfo"], + billingInfo:ValueTypes["BuyerInfo"], + token?:string +}; + ["CheckoutInput"]: { + contactInfo:ValueTypes["BuyerInfo"], + billingInfo:ValueTypes["BuyerInfo"], + token?:string +}; + ["CountryCodesComponentEnum"]:CountryCodesComponentEnum; + ["CurrencyEnum"]:CurrencyEnum; + ["DateExceptionComponent"]: AliasType<{ + dates?:ValueTypes["DateRangeComponent"], + hours?:ValueTypes["HourComponent"], + allDay?:true, + __typename?: true +}>; + ["DateExceptionComponentInput"]: { + dates:ValueTypes["DateRangeComponentInput"], + hours?:ValueTypes["HoursInput"][], + allDay?:boolean +}; + ["DateRangeComponent"]: AliasType<{ + startDate?:true, + endDate?:true, + __typename?: true +}>; + ["DateRangeComponentInput"]: { + startDate:ValueTypes["DateTime"], + endDate:ValueTypes["DateTime"] +}; + ["DateTabType"]:DateTabType; + /** The javascript `Date` as string. Type represents date and time as the ISO Date string. */ +["DateTime"]:unknown; + ["DurationTypeEnum"]:DurationTypeEnum; + ["EditAccountInput"]: { + userName?:string, + firstName?:string, + lastName?:string, + profilePicture?:ValueTypes["ImageInput"], + phoneNumber?:string, + gender?:ValueTypes["AccountGenderEnum"], + birthDate?:ValueTypes["DateTime"], + address?:ValueTypes["AddressInput"], + language?:ValueTypes["AvailableTranslation"], + settings?:ValueTypes["SettingsInput"][] +}; + ["EditMessageInput"]: { + message:string +}; + ["EditWorkspaceInput"]: { + title?:string, + teaser?:string, + cover?:ValueTypes["ImageInput"], + thumbnail?:ValueTypes["ImageInput"], + extraImages?:ValueTypes["ImageInput"][], + content?:string, + seo?:ValueTypes["SEOSimpleInput"], + urls?:string, + workspaceType?:ValueTypes["WorkspaceTypeEnum"], + pricingPerHour?:number, + currency:ValueTypes["AvailableCurrency"], + maxCapacity?:number, + minStay?:number, + maxStay?:number, + availability?:ValueTypes["AvailabilityInput"], + mainImages:ValueTypes["MainImage"], + equipmentIds?:string[], + featureIds?:string[], + address?:ValueTypes["AddressStrictInput"] +}; + ["EnginePathComponent"]: AliasType<{ + ressourceModel?:true, + ressourceId?:true, + __typename?: true +}>; + ["FavoriteComponent"]: AliasType<{ + ressourceId?:true, + ressourceType?:true, + addedAt?:true, + favorite?:true, + liked?:true, + __typename?: true +}>; + ["FirebaseTokenResult"]: AliasType<{ + localId?:true, + email?:true, + displayName?:true, + idToken?:true, + registered?:true, + refreshToken?:true, + expiresIn?:true, + __typename?: true +}>; + ["GeolocAddressSearchInput"]: { + formattedAddress:string, + radius?:number +}; + ["GeolocDistComponent"]: AliasType<{ + calculated?:true, + location?:ValueTypes["LocComponent"], + __typename?: true +}>; + ["GeolocSearchInput"]: { + longitude:number, + latitude:number, + radius?:number +}; + ["GetArgs"]: { + limit:number, + skip:number, + sort?:string +}; + ["HourComponent"]: AliasType<{ + from?:true, + to?:true, + __typename?: true +}>; + ["HoursInput"]: { + from:string, + to:string +}; + ["IEngineSchema"]:AliasType<{ + _id?:true, + organisationId?:true, + paths?:ValueTypes["EnginePathComponent"], + by?:ValueTypes["MetaBy"], + permissions?:ValueTypes["MetaPermissions"], + createdAt?:true, + updatedAt?:true; + ['...on BookingEngineSchema']?: Omit; + ['...on BookingSchema']?: Omit; + ['...on OrderEngineSchema']?: Omit; + __typename?: true +}>; + ["ImageComponent"]: AliasType<{ + title?:true, + fileType?:true, + large?:true, + medium?:true, + small?:true, + __typename?: true +}>; + ["ImageInput"]: { + title?:string, + fileType?:string, + large:string, + medium?:string, + small?:string +}; + ["InvoiceInfo"]: AliasType<{ + provider?:true, + invoiceId?:true, + invoiceNumber?:true, + __typename?: true +}>; + ["InvoicingProvider"]:InvoicingProvider; + /** The `JSONObject` scalar type represents JSON objects as specified by [ECMA-404](http://www.ecma-international.org/publications/files/ECMA-ST/ECMA-404.pdf). */ +["JSONObject"]:unknown; + ["LineItemSchema"]: AliasType<{ + _id?:true, + productRessource?:true, + productId?:true, + title?:true, + price?:true, + salesPrice?:true, + quantity?:true, + vatClassId?:true, + parentId?:true, + addOns?:ValueTypes["AddOnComponent"], + localeInfo?:ValueTypes["LocaleInfo"], + promoId?:true, + finalPrice?:true, + getProduct?:ValueTypes["ProductUnion"], + getLinePrice?:true, + getLineVatPrice?:true, + __typename?: true +}>; + ["LinkEmailInput"]: { + email:string, + password:string, + idToken:string +}; + ["ListEnum"]:ListEnum; + ["ListModel"]: AliasType<{ + _id?:true, + organisationId?:true, + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + createdAt?:true, + updatedAt?:true, + r?:true, + w?:true, + d?:true, + title?:ValueTypes["TranslatableComponent"], + teaser?:ValueTypes["TranslatableComponent"], + cover?:ValueTypes["ImageComponent"], + thumbnail?:ValueTypes["ImageComponent"], + extraImages?:ValueTypes["ImageComponent"], + content?:ValueTypes["TranslatableComponent"], + seo?:ValueTypes["SEOField"], + urls?:ValueTypes["TranslatableComponent"], + ressourceType?:true, + colorCode?:true, + __typename?: true +}>; + ["LocaleInfo"]: AliasType<{ + countryCode?:true, + locale?:true, + __typename?: true +}>; + ["LocComponent"]: AliasType<{ + type?:true, + coordinates?:true, + __typename?: true +}>; + ["LoginInput"]: { + email:string, + password:string +}; + ["MainImage"]: { + main?:ValueTypes["ImageInput"], + one?:ValueTypes["ImageInput"], + two?:ValueTypes["ImageInput"], + three?:ValueTypes["ImageInput"], + four?:ValueTypes["ImageInput"] +}; + ["MainImageComponent"]: AliasType<{ + main?:ValueTypes["ImageComponent"], + one?:ValueTypes["ImageComponent"], + two?:ValueTypes["ImageComponent"], + three?:ValueTypes["ImageComponent"], + four?:ValueTypes["ImageComponent"], + __typename?: true +}>; + ["MessageModel"]: AliasType<{ + _id?:true, + organisationId?:true, + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + createdAt?:true, + updatedAt?:true, + r?:true, + w?:true, + d?:true, + roomId?:true, + notified?:true, + message?:true, + edited?:true, + deleted?:true, + file?:ValueTypes["ImageComponent"], + sentBy?:true, + readBy?:ValueTypes["MessageReadBy"], + __typename?: true +}>; + ["MessageReadBy"]: AliasType<{ + accountId?:true, + readAt?:true, + __typename?: true +}>; + ["MetaBy"]: AliasType<{ + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + __typename?: true +}>; + ["MetaPermissions"]: AliasType<{ + r?:true, + w?:true, + d?:true, + __typename?: true +}>; + ["ModelLoadersEnum"]:ModelLoadersEnum; + ["Mutation"]: AliasType<{ +updateMe?: [{ input:ValueTypes["EditAccountInput"]},ValueTypes["AccountModel"]], +updateMeEmail?: [{ input:ValueTypes["NewEmailInput"]},ValueTypes["AccountModel"]], +updateMePassword?: [{ input:ValueTypes["NewPasswordInput"]},ValueTypes["AccountModel"]], +resetPassword?: [{ input:ValueTypes["ResetPasswordInput"]},ValueTypes["SimpleResult"]], +registerGuest?: [{ otherInfo:ValueTypes["EditAccountInput"], input:ValueTypes["LinkEmailInput"]},ValueTypes["AccountModel"]], +register?: [{ input:ValueTypes["NewAccountInput"]},ValueTypes["AccountModel"]], +accountsBillingInfosAddOne?: [{ input:ValueTypes["BillingAddressInput"]},ValueTypes["BillingAddressComponent"]], +accountsBillingInfosEditOne?: [{ input:ValueTypes["BillingAddressInput"], id:string},ValueTypes["BillingAddressComponent"]], +accountsBillingInfosDeleteOne?: [{ id:string},true], +accountsPaymentMethodsAddOne?: [{ input:ValueTypes["PaymentMethodInput"]},ValueTypes["PaymentMethodDetail"]], +accountsPaymentMethodsMarkasDefault?: [{ id:string},ValueTypes["PaymentMethodDetail"]], +accountsPaymentMethodsDeleteOne?: [{ id:string},true], + accountsLinkWithStripe?:true, + accountsRefreshInfoStripe?:ValueTypes["AccountModel"], +accountsFavoritesAddOne?: [{ input:ValueTypes["AddInput"]},ValueTypes["FavoriteComponent"]], +accountsFavoritesRemoveOne?: [{ input:ValueTypes["RemoveInput"]},ValueTypes["SuccessResponse"]], +workspacesAddOne?: [{ input:ValueTypes["NewWorkspaceInput"]},ValueTypes["WorkspaceModel"]], +workspacesEditOne?: [{ input:ValueTypes["EditWorkspaceInput"], id:string},ValueTypes["WorkspaceModel"]], +workspacesDeleteOne?: [{ id:string},ValueTypes["WorkspaceModel"]], +finaliseCheckoutWithStripe?: [{ input:ValueTypes["CheckoutInput"]},ValueTypes["OrderEngineSchema"]], + removeCurrentOrder?:ValueTypes["OrderEngineSchema"], +validateOrder?: [{ id:string},ValueTypes["OrderEngineSchema"]], +roomsCreateOne?: [{ input:ValueTypes["NewRoomInput"]},ValueTypes["RoomModel"]], +roomsAddPeople?: [{ input:ValueTypes["AddPeopleToRoomInput"]},ValueTypes["RoomModel"]], +roomsRemovePeople?: [{ input:ValueTypes["RemovePeopleFromRoomInput"]},ValueTypes["RoomModel"]], +roomsMarkAllMessageAsRead?: [{ roomId:string},ValueTypes["RoomModel"]], +reserveOneBooking?: [{ input:ValueTypes["BookingNewInputSchema"]},ValueTypes["OrderEngineSchema"]], +roomsAddOneMessage?: [{ input:ValueTypes["NewMessageInput"]},ValueTypes["MessageModel"]], +roomsEditOneMessage?: [{ input:ValueTypes["EditMessageInput"], id:string},ValueTypes["MessageModel"]], +roomsMarkMessageAsRead?: [{ id:string},ValueTypes["MessageModel"]], +roomsDeleteOneMessage?: [{ id:string},ValueTypes["MessageModel"]], + __typename?: true +}>; + ["NewAccountInput"]: { + email:string, + password:string, + passwordConfirmation:string, + userName?:string, + firstName?:string, + lastName?:string, + profilePicture?:ValueTypes["ImageInput"], + phoneNumber?:string, + gender?:ValueTypes["AccountGenderEnum"], + birthDate?:ValueTypes["DateTime"], + address?:ValueTypes["AddressInput"], + language?:ValueTypes["AvailableTranslation"], + settings?:ValueTypes["SettingsInput"][], + terms:boolean +}; + ["NewEmailInput"]: { + newEmail:string +}; + ["NewMessageInput"]: { + roomId:string, + message:string, + file?:ValueTypes["ImageInput"] +}; + ["NewPasswordInput"]: { + oldPassword:string, + newPassword:string, + newPasswordConfirmation:string +}; + ["NewRoomInput"]: { + accountIds:string[] +}; + ["NewWorkspaceInput"]: { + title?:string, + teaser?:string, + cover?:ValueTypes["ImageInput"], + thumbnail?:ValueTypes["ImageInput"], + extraImages?:ValueTypes["ImageInput"][], + content?:string, + seo?:ValueTypes["SEOSimpleInput"], + urls?:string, + workspaceType:ValueTypes["WorkspaceTypeEnum"], + pricingPerHour:number, + currency:ValueTypes["AvailableCurrency"], + maxCapacity:number, + minStay:number, + maxStay?:number, + availability:ValueTypes["AvailabilityInput"], + mainImages:ValueTypes["MainImage"], + equipmentIds:string[], + featureIds:string[], + address:ValueTypes["AddressStrictInput"] +}; + ["OrderBusinessStatusEnum"]:OrderBusinessStatusEnum; + ["OrderEngineSchema"]: AliasType<{ + lines?:ValueTypes["LineItemSchema"], + localeInfo?:ValueTypes["LocaleInfo"], + currency?:true, + vatExempt?:true, + accountId?:true, + contactInfo?:ValueTypes["BuyerInfoComponent"], + billingInfo?:ValueTypes["BuyerInfoComponent"], + promoId?:true, + invoiceInfo?:ValueTypes["InvoiceInfo"], + providerOrderItems?:ValueTypes["ProviderSchema"], + orderName?:ValueTypes["TranslatableComponent"], + orderStatus?:true, + paymentIntent?:ValueTypes["PaymentIntentInfo"], + paymentStatus?:true, + paymentInfo?:ValueTypes["PaymentInfo"], + subTotalPrice?:true, + vatPrice?:true, + finalPrice?:true, + promo?:ValueTypes["PromoModel"], + _id?:true, + organisationId?:true, + paths?:ValueTypes["EnginePathComponent"], + by?:ValueTypes["MetaBy"], + permissions?:ValueTypes["MetaPermissions"], + createdAt?:true, + updatedAt?:true, + getBookingInput?:ValueTypes["BookingNewInput"], + __typename?: true +}>; + ["OrderStatusEnum"]:OrderStatusEnum; + ["PaymentInfo"]: AliasType<{ + provider?:true, + transactionId?:true, + __typename?: true +}>; + ["PaymentIntentInfo"]: AliasType<{ + provider?:true, + stripePaymentIntentData?:ValueTypes["StripePaymentIntentData"], + __typename?: true +}>; + ["PaymentMethodDetail"]: AliasType<{ + _id?:true, + sPayMethodId?:true, + default?:true, + nameOnCard?:true, + type?:true, + cardInfo?:ValueTypes["CardInfo"], + __typename?: true +}>; + ["PaymentMethodInput"]: { + sPayMethodId:string, + nameOnCard:string, + default?:boolean +}; + ["PaymentProvider"]:PaymentProvider; + ["PaymentStatusEnum"]:PaymentStatusEnum; + ["PlaceComponent"]: AliasType<{ + placeId?:true, + address?:ValueTypes["AddressComponent"], + loc?:ValueTypes["LocComponent"], + formattedAddress?:true, + __typename?: true +}>; + ["ProductRessourceEnum"]:ProductRessourceEnum; + ["ProductUnion"]: AliasType<{ ["...on WorkspaceModel"] : ValueTypes["WorkspaceModel"] + __typename?: true +}>; + ["PromoModel"]: AliasType<{ + _id?:true, + organisationId?:true, + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + createdAt?:true, + updatedAt?:true, + r?:true, + w?:true, + d?:true, + code?:true, + description?:ValueTypes["TranslatableComponent"], + type?:true, + value?:true, + validity?:true, + cummulable?:true, + usageLimit?:true, + usage?:true, + __typename?: true +}>; + ["PromoType"]:PromoType; + ["ProviderSchema"]: AliasType<{ + _id?:true, + organisationId?:true, + orderStatus?:true, + lines?:ValueTypes["LineItemSchema"], + __typename?: true +}>; + ["Query"]: AliasType<{ + me?:ValueTypes["AccountModel"], +login?: [{ refreshToken?:string, creds?:ValueTypes["LoginInput"]},ValueTypes["FirebaseTokenResult"]], +magicLink?: [{ input:ValueTypes["ResetPasswordInput"]},ValueTypes["SimpleResult"]], +placesAutocomplete?: [{ session:string, input:string, language?:ValueTypes["AvailableTranslation"], country?:ValueTypes["CountryCodesComponentEnum"][]},true], +placesGeocode?: [{ session:string, placeId:string},true], +placesGeocodeFromAddress?: [{ address:string},true], +categoriesGetOne?: [{ id:string},ValueTypes["CategoryModel"]], +categoriesGetMany?: [{ search?:string, ressourceType:ValueTypes["RessourceEnum"], pagination?:ValueTypes["GetArgs"]},ValueTypes["CategoryModel"]], +categoriesGetCount?: [{ search?:string, ressourceType:ValueTypes["RessourceEnum"], pagination?:ValueTypes["GetArgs"]},true], +listsGetOne?: [{ id:string},ValueTypes["ListModel"]], +listsGetMany?: [{ search?:string, ressourceType:ValueTypes["ListEnum"], pagination?:ValueTypes["GetArgs"]},ValueTypes["ListModel"]], +listsGetCount?: [{ search?:string, ressourceType:ValueTypes["ListEnum"], pagination?:ValueTypes["GetArgs"]},true], + accountsDashboardLinkStripe?:true, +workspacesGetOne?: [{ id:string},ValueTypes["WorkspaceModel"]], +workspacesGetMany?: [{ geoSearch?:ValueTypes["GeolocSearchInput"], geoSearchWithAddress?:ValueTypes["GeolocAddressSearchInput"], _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"], dateFrom?:ValueTypes["DateTime"], dateTo?:ValueTypes["DateTime"], hoursFrom?:string, hoursTo?:string, workspaceType?:ValueTypes["WorkspaceTypeEnum"], workspaceTypes?:ValueTypes["WorkspaceTypeEnum"][], priceMin?:number, priceMax?:number, maxCapacity?:number, equipmentIds?:string[], featureIds?:string[]},ValueTypes["WorkspaceModel"]], +workspacesGetCount?: [{ geoSearch?:ValueTypes["GeolocSearchInput"], geoSearchWithAddress?:ValueTypes["GeolocAddressSearchInput"], _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"], dateFrom?:ValueTypes["DateTime"], dateTo?:ValueTypes["DateTime"], hoursFrom?:string, hoursTo?:string, workspaceType?:ValueTypes["WorkspaceTypeEnum"], workspaceTypes?:ValueTypes["WorkspaceTypeEnum"][], priceMin?:number, priceMax?:number, maxCapacity?:number, equipmentIds?:string[], featureIds?:string[]},true], +workspacesSearchMany?: [{ geoSearch?:ValueTypes["GeolocSearchInput"], geoSearchWithAddress?:ValueTypes["GeolocAddressSearchInput"], _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"], dateFrom?:ValueTypes["DateTime"], dateTo?:ValueTypes["DateTime"], hoursFrom?:string, hoursTo?:string, workspaceType?:ValueTypes["WorkspaceTypeEnum"], workspaceTypes?:ValueTypes["WorkspaceTypeEnum"][], priceMin?:number, priceMax?:number, maxCapacity?:number, equipmentIds?:string[], featureIds?:string[]},ValueTypes["WorkspaceModel"]], +workspacesGetMine?: [{ geoSearch?:ValueTypes["GeolocSearchInput"], geoSearchWithAddress?:ValueTypes["GeolocAddressSearchInput"], _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"], dateFrom?:ValueTypes["DateTime"], dateTo?:ValueTypes["DateTime"], hoursFrom?:string, hoursTo?:string, workspaceType?:ValueTypes["WorkspaceTypeEnum"], workspaceTypes?:ValueTypes["WorkspaceTypeEnum"][], priceMin?:number, priceMax?:number, maxCapacity?:number, equipmentIds?:string[], featureIds?:string[]},ValueTypes["WorkspaceModel"]], +workspacesGetMineCount?: [{ geoSearch?:ValueTypes["GeolocSearchInput"], geoSearchWithAddress?:ValueTypes["GeolocAddressSearchInput"], _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"], dateFrom?:ValueTypes["DateTime"], dateTo?:ValueTypes["DateTime"], hoursFrom?:string, hoursTo?:string, workspaceType?:ValueTypes["WorkspaceTypeEnum"], workspaceTypes?:ValueTypes["WorkspaceTypeEnum"][], priceMin?:number, priceMax?:number, maxCapacity?:number, equipmentIds?:string[], featureIds?:string[]},true], +bookingsGetOne?: [{ id:string},ValueTypes["BookingEngineSchema"]], +bookingsGetMany?: [{ startDate?:ValueTypes["DateTime"], endDate?:ValueTypes["DateTime"], rangeType?:ValueTypes["RangeType"], dateTabType?:ValueTypes["DateTabType"], dateRange?:ValueTypes["DateRangeComponentInput"], _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"], ressourceModel?:ValueTypes["BookingsRessourceEnum"], ressourceId?:string, status?:ValueTypes["StatusEnum"][]},ValueTypes["BookingEngineSchema"]], +bookingsGetCount?: [{ startDate?:ValueTypes["DateTime"], endDate?:ValueTypes["DateTime"], rangeType?:ValueTypes["RangeType"], dateTabType?:ValueTypes["DateTabType"], dateRange?:ValueTypes["DateRangeComponentInput"], _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"], ressourceModel?:ValueTypes["BookingsRessourceEnum"], ressourceId?:string, status?:ValueTypes["StatusEnum"][]},true], + myCurrentOrder?:ValueTypes["OrderEngineSchema"], +myOrdersGetOne?: [{ id:string},ValueTypes["OrderEngineSchema"]], +myOrdersGetMany?: [{ orderStatus?:ValueTypes["OrderStatusEnum"], paymentStatus?:ValueTypes["PaymentStatusEnum"], pagination?:ValueTypes["GetArgs"], search?:string},ValueTypes["OrderEngineSchema"]], +myOrdersGetManyCount?: [{ orderStatus?:ValueTypes["OrderStatusEnum"], paymentStatus?:ValueTypes["PaymentStatusEnum"], pagination?:ValueTypes["GetArgs"], search?:string},true], +roomsGetOne?: [{ id:string},ValueTypes["RoomModel"]], +roomsGetMany?: [{ pagination?:ValueTypes["GetArgs"]},ValueTypes["RoomModel"]], +roomsGetMessages?: [{ pagination?:ValueTypes["GetArgs"], id:string},ValueTypes["MessageModel"]], + __typename?: true +}>; + ["RangeType"]:RangeType; + ["RemoveInput"]: { + ressourceId:string +}; + ["RemovePeopleFromRoomInput"]: { + roomId:string, + accountIds:string[] +}; + ["ResetPasswordInput"]: { + email:string +}; + ["RessourceEnum"]:RessourceEnum; + ["RoomModel"]: AliasType<{ + _id?:true, + organisationId?:true, + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + createdAt?:true, + updatedAt?:true, + r?:true, + w?:true, + d?:true, + title?:true, + stats?:ValueTypes["RoomsStats"], + getAccounts?:ValueTypes["AccountModel"], +getMessages?: [{ pagination?:ValueTypes["GetArgs"]},ValueTypes["MessageModel"]], + lastMessage?:true, + __typename?: true +}>; + ["RoomsStats"]: AliasType<{ + unreadCounts?:ValueTypes["UnreadCount"], + __typename?: true +}>; + ["SEOField"]: AliasType<{ + title?:ValueTypes["TranslatableComponent"], + description?:ValueTypes["TranslatableComponent"], + keywords?:ValueTypes["TranslatableComponent"], + thumbnail?:ValueTypes["ImageComponent"], + __typename?: true +}>; + ["SEOSimpleField"]: AliasType<{ + title?:true, + description?:true, + keywords?:true, + thumbnail?:ValueTypes["ImageComponent"], + __typename?: true +}>; + ["SEOSimpleInput"]: { + title:string, + description:string, + keywords?:string, + thumbnail?:ValueTypes["ImageInput"] +}; + ["SettingsComponent"]: AliasType<{ + type?:true, + email?:true, + pushNotifications?:true, + sms?:true, + __typename?: true +}>; + ["SettingsInput"]: { + type:ValueTypes["SettingsTypeEnum"], + email?:boolean, + pushNotifications?:boolean, + sms?:boolean +}; + ["SettingsTypeEnum"]:SettingsTypeEnum; + ["SimpleResult"]: AliasType<{ + message?:true, + __typename?: true +}>; + ["SlotComponent"]: AliasType<{ + startDate?:true, + endDate?:true, + startTime?:true, + endTime?:true, + __typename?: true +}>; + ["SlotComponentInput"]: { + startDate:ValueTypes["DateTime"], + endDate:ValueTypes["DateTime"], + startTime:string, + endTime:string +}; + ["StatusEnum"]:StatusEnum; + ["StripeInfo"]: AliasType<{ + customerId?:true, + __typename?: true +}>; + ["StripePaymentIntentData"]: AliasType<{ + id?:true, + client_secret?:true, + currency?:true, + customer?:true, + status?:true, + __typename?: true +}>; + ["StripePayoutInfo"]: AliasType<{ + accountId?:true, + chargesEnabled?:true, + payoutsEnabled?:true, + detailsSubmitted?:true, + __typename?: true +}>; + ["SuccessResponse"]: AliasType<{ + success?:true, + __typename?: true +}>; + ["TranslatableComponent"]: AliasType<{ + en?:true, + fr?:true, + nl?:true, + de?:true, + __typename?: true +}>; + ["UnreadCount"]: AliasType<{ + accountId?:true, + count?:true, + __typename?: true +}>; + ["WorkspaceModel"]: AliasType<{ + _id?:true, + organisationId?:true, + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + createdAt?:true, + updatedAt?:true, + r?:true, + w?:true, + d?:true, + title?:true, + teaser?:true, + cover?:ValueTypes["ImageComponent"], + thumbnail?:ValueTypes["ImageComponent"], + extraImages?:ValueTypes["ImageComponent"], + content?:true, + seo?:ValueTypes["SEOSimpleField"], + urls?:true, + place?:ValueTypes["PlaceComponent"], + dist?:ValueTypes["GeolocDistComponent"], +getDistFromLocation?: [{ longitude:number, latitude:number, radius?:number},true], + workspaceType?:true, + pricingPerHour?:true, + currency?:true, + maxCapacity?:true, + minStay?:true, + maxStay?:true, + availability?:ValueTypes["AvailabilityComponent"], + mainImages?:ValueTypes["MainImageComponent"], + equipmentIds?:true, + featureIds?:true, + address?:ValueTypes["AddressComponent"], + finalPrice?:true, + getEquipments?:ValueTypes["ListModel"], + getFeatures?:ValueTypes["ListModel"], +getBookings?: [{ ressourceModel?:ValueTypes["BookingsRessourceEnum"], ressourceId?:string, status?:ValueTypes["StatusEnum"][], startDate?:ValueTypes["DateTime"], endDate?:ValueTypes["DateTime"], rangeType?:ValueTypes["RangeType"], dateTabType?:ValueTypes["DateTabType"], dateRange?:ValueTypes["DateRangeComponentInput"], _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"]},ValueTypes["BookingSchema"]], + getOwner?:ValueTypes["WorkspaceOwnerComponent"], + __typename?: true +}>; + ["WorkspaceOwnerComponent"]: AliasType<{ + firstName?:true, + lastName?:true, + __typename?: true +}>; + ["WorkspaceTypeEnum"]:WorkspaceTypeEnum + } + +export type PartialObjects = { + ["AccountGenderEnum"]:AccountGenderEnum, + ["AccountModel"]: { + __typename?: "AccountModel"; + _id?:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt?:PartialObjects["DateTime"], + updatedAt?:PartialObjects["DateTime"], + r?:string[], + w?:string[], + d?:string[], + types?:PartialObjects["AccountTypeEnum"][], + organisationIds?:string[], + email?:string, + userName?:string, + firstName?:string, + lastName?:string, + profilePicture?:PartialObjects["ImageComponent"], + phoneNumber?:string, + gender?:PartialObjects["AccountGenderEnum"], + birthDate?:PartialObjects["DateTime"], + getAge?:number, + address?:PartialObjects["AddressComponent"], + language?:PartialObjects["AvailableTranslation"], + settings?:PartialObjects["SettingsComponent"][], + paymentInfo?:PartialObjects["AccountPaymentInfo"], + payoutInfo?:PartialObjects["AccountPayoutInfo"], + terms?:boolean, + favLikes?:(PartialObjects["FavoriteComponent"] | undefined)[], + getFavorites?:PartialObjects["FavoriteComponent"][], + getLiked?:PartialObjects["FavoriteComponent"][] + }, + ["AccountPaymentInfo"]: { + __typename?: "AccountPaymentInfo"; + stripeInfo?:PartialObjects["StripeInfo"], + paymentMethods?:PartialObjects["PaymentMethodDetail"][], + billingInfos?:PartialObjects["BillingAddressComponent"][] + }, + ["AccountPayoutInfo"]: { + __typename?: "AccountPayoutInfo"; + stripeInfo?:PartialObjects["StripePayoutInfo"] + }, + ["AccountTypeEnum"]:AccountTypeEnum, + ["AddInput"]: { + ressourceId:string, + ressourceType:PartialObjects["RessourceEnum"] +}, + ["AddOnComponent"]: { + __typename?: "AddOnComponent"; + addOnId?:string, + quantity?:number + }, + ["AddPeopleToRoomInput"]: { + roomId:string, + accountIds:string[] +}, + ["AddressComponent"]: { + __typename?: "AddressComponent"; + number?:string, + street?:string, + streetBis?:string, + floor?:string, + box?:string, + zip?:string, + state?:string, + city?:string, + country?:string + }, + ["AddressInput"]: { + number?:string, + street?:string, + streetBis?:string, + floor?:string, + box?:string, + zip?:string, + state?:string, + city:string, + country:string +}, + ["AddressStrictComponent"]: { + __typename?: "AddressStrictComponent"; + number?:string, + street?:string, + zip?:string, + city?:string, + country?:string, + streetBis?:string, + floor?:string, + box?:string, + state?:string + }, + ["AddressStrictInput"]: { + number:string, + street:string, + zip:string, + city:string, + country:string, + streetBis?:string, + floor?:string, + box?:string, + state?:string +}, + ["AvailabilityComponent"]: { + __typename?: "AvailabilityComponent"; + dates?:PartialObjects["DateRangeComponent"], + hours?:PartialObjects["HourComponent"][], + exceptions?:PartialObjects["DateExceptionComponent"][], + noWeekend?:boolean + }, + ["AvailabilityInput"]: { + dates:PartialObjects["DateRangeComponentInput"], + hours?:PartialObjects["HoursInput"][], + exceptions?:PartialObjects["DateExceptionComponentInput"][], + noWeekend?:boolean +}, + ["AvailableCurrency"]:AvailableCurrency, + ["AvailableTranslation"]:AvailableTranslation, + ["BillingAddressComponent"]: { + __typename?: "BillingAddressComponent"; + firstName?:string, + lastName?:string, + address?:PartialObjects["AddressStrictComponent"], + email?:string, + default?:boolean, + customTags?:PartialObjects["JSONObject"], + company?:string, + vatNumber?:string, + fiscalForm?:string, + _id?:string + }, + ["BillingAddressInput"]: { + firstName:string, + lastName:string, + address:PartialObjects["AddressStrictInput"], + email?:string, + default?:boolean, + customTags?:PartialObjects["JSONObject"], + company?:string, + vatNumber?:string, + fiscalForm?:string +}, + ["BookingEngineSchema"]: { + __typename?: "BookingEngineSchema"; + durationType?:PartialObjects["DurationTypeEnum"], + startToEnd?:PartialObjects["DateRangeComponent"], + slot?:PartialObjects["SlotComponent"], + capacity?:number, + comments?:string, + status?:PartialObjects["StatusEnum"], + paths?:PartialObjects["EnginePathComponent"][], + ownerId?:string, + dates?:PartialObjects["DateRangeComponent"][], + owner?:PartialObjects["AccountModel"], + _id?:string, + organisationId?:string, + by?:PartialObjects["MetaBy"], + permissions?:PartialObjects["MetaPermissions"], + createdAt?:PartialObjects["DateTime"], + updatedAt?:PartialObjects["DateTime"], + workspace?:PartialObjects["WorkspaceModel"], + room?:PartialObjects["RoomModel"] + }, + ["BookingNewInput"]: { + __typename?: "BookingNewInput"; + durationType?:PartialObjects["DurationTypeEnum"], + startToEnd?:PartialObjects["DateRangeComponent"], + slot?:PartialObjects["SlotComponent"], + capacity?:number, + comments?:string, + ressourceModel?:PartialObjects["BookingsRessourceEnum"], + ressourceId?:string, + checkoutInfo?:PartialObjects["CheckoutEngineInput"] + }, + ["BookingNewInputSchema"]: { + durationType:PartialObjects["DurationTypeEnum"], + startToEnd?:PartialObjects["DateRangeComponentInput"], + slot?:PartialObjects["SlotComponentInput"], + capacity?:number, + comments?:string, + ressourceModel:PartialObjects["BookingsRessourceEnum"], + ressourceId:string, + checkoutInfo?:PartialObjects["CheckoutEngineInputI"] +}, + ["BookingSchema"]: { + __typename?: "BookingSchema"; + durationType?:PartialObjects["DurationTypeEnum"], + startToEnd?:PartialObjects["DateRangeComponent"], + slot?:PartialObjects["SlotComponent"], + capacity?:number, + comments?:string, + status?:PartialObjects["StatusEnum"], + paths?:PartialObjects["EnginePathComponent"][], + ownerId?:string, + dates?:PartialObjects["DateRangeComponent"][], + _id?:string, + organisationId?:string, + by?:PartialObjects["MetaBy"], + permissions?:PartialObjects["MetaPermissions"], + createdAt?:PartialObjects["DateTime"], + updatedAt?:PartialObjects["DateTime"] + }, + ["BookingsRessourceEnum"]:BookingsRessourceEnum, + ["BuyerInfo"]: { + firstName:string, + lastName:string, + email:string, + company?:string, + vatNumber?:string, + address:PartialObjects["AddressInput"] +}, + ["BuyerInfoComponent"]: { + __typename?: "BuyerInfoComponent"; + firstName?:string, + lastName?:string, + email?:string, + company?:string, + vatNumber?:string, + address?:PartialObjects["AddressComponent"] + }, + ["cardBrandEnum"]:cardBrandEnum, + ["CardInfo"]: { + __typename?: "CardInfo"; + brand?:PartialObjects["cardBrandEnum"], + country?:string, + exp_month?:number, + exp_year?:number, + last4?:string + }, + ["cardTypeEnum"]:cardTypeEnum, + ["CategoryModel"]: { + __typename?: "CategoryModel"; + _id?:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt?:PartialObjects["DateTime"], + updatedAt?:PartialObjects["DateTime"], + r?:string[], + w?:string[], + d?:string[], + title?:PartialObjects["TranslatableComponent"], + teaser?:PartialObjects["TranslatableComponent"], + cover?:PartialObjects["ImageComponent"], + thumbnail?:PartialObjects["ImageComponent"], + extraImages?:PartialObjects["ImageComponent"][], + content?:PartialObjects["TranslatableComponent"], + seo?:PartialObjects["SEOField"], + urls?:PartialObjects["TranslatableComponent"], + ressourceType?:PartialObjects["RessourceEnum"], + colorCode?:string + }, + ["CheckoutEngineInput"]: { + __typename?: "CheckoutEngineInput"; + contactInfo?:PartialObjects["BuyerInfoComponent"], + billingInfo?:PartialObjects["BuyerInfoComponent"], + token?:string + }, + ["CheckoutEngineInputI"]: { + contactInfo:PartialObjects["BuyerInfo"], + billingInfo:PartialObjects["BuyerInfo"], + token?:string +}, + ["CheckoutInput"]: { + contactInfo:PartialObjects["BuyerInfo"], + billingInfo:PartialObjects["BuyerInfo"], + token?:string +}, + ["CountryCodesComponentEnum"]:CountryCodesComponentEnum, + ["CurrencyEnum"]:CurrencyEnum, + ["DateExceptionComponent"]: { + __typename?: "DateExceptionComponent"; + dates?:PartialObjects["DateRangeComponent"], + hours?:PartialObjects["HourComponent"][], + allDay?:boolean + }, + ["DateExceptionComponentInput"]: { + dates:PartialObjects["DateRangeComponentInput"], + hours?:PartialObjects["HoursInput"][], + allDay?:boolean +}, + ["DateRangeComponent"]: { + __typename?: "DateRangeComponent"; + startDate?:PartialObjects["DateTime"], + endDate?:PartialObjects["DateTime"] + }, + ["DateRangeComponentInput"]: { + startDate:PartialObjects["DateTime"], + endDate:PartialObjects["DateTime"] +}, + ["DateTabType"]:DateTabType, + /** The javascript `Date` as string. Type represents date and time as the ISO Date string. */ +["DateTime"]:any, + ["DurationTypeEnum"]:DurationTypeEnum, + ["EditAccountInput"]: { + userName?:string, + firstName?:string, + lastName?:string, + profilePicture?:PartialObjects["ImageInput"], + phoneNumber?:string, + gender?:PartialObjects["AccountGenderEnum"], + birthDate?:PartialObjects["DateTime"], + address?:PartialObjects["AddressInput"], + language?:PartialObjects["AvailableTranslation"], + settings?:PartialObjects["SettingsInput"][] +}, + ["EditMessageInput"]: { + message:string +}, + ["EditWorkspaceInput"]: { + title?:string, + teaser?:string, + cover?:PartialObjects["ImageInput"], + thumbnail?:PartialObjects["ImageInput"], + extraImages?:PartialObjects["ImageInput"][], + content?:string, + seo?:PartialObjects["SEOSimpleInput"], + urls?:string, + workspaceType?:PartialObjects["WorkspaceTypeEnum"], + pricingPerHour?:number, + currency:PartialObjects["AvailableCurrency"], + maxCapacity?:number, + minStay?:number, + maxStay?:number, + availability?:PartialObjects["AvailabilityInput"], + mainImages:PartialObjects["MainImage"], + equipmentIds?:string[], + featureIds?:string[], + address?:PartialObjects["AddressStrictInput"] +}, + ["EnginePathComponent"]: { + __typename?: "EnginePathComponent"; + ressourceModel?:PartialObjects["ModelLoadersEnum"], + ressourceId?:string + }, + ["FavoriteComponent"]: { + __typename?: "FavoriteComponent"; + ressourceId?:string, + ressourceType?:PartialObjects["RessourceEnum"], + addedAt?:PartialObjects["DateTime"], + favorite?:boolean, + liked?:boolean + }, + ["FirebaseTokenResult"]: { + __typename?: "FirebaseTokenResult"; + localId?:string, + email?:string, + displayName?:string, + idToken?:string, + registered?:boolean, + refreshToken?:string, + expiresIn?:string + }, + ["GeolocAddressSearchInput"]: { + formattedAddress:string, + radius?:number +}, + ["GeolocDistComponent"]: { + __typename?: "GeolocDistComponent"; + calculated?:number, + location?:PartialObjects["LocComponent"] + }, + ["GeolocSearchInput"]: { + longitude:number, + latitude:number, + radius?:number +}, + ["GetArgs"]: { + limit:number, + skip:number, + sort?:string +}, + ["HourComponent"]: { + __typename?: "HourComponent"; + from?:string, + to?:string + }, + ["HoursInput"]: { + from:string, + to:string +}, + ["IEngineSchema"]:{ + _id?:string; + organisationId?:string; + paths?:PartialObjects["EnginePathComponent"][]; + by?:PartialObjects["MetaBy"]; + permissions?:PartialObjects["MetaPermissions"]; + createdAt?:PartialObjects["DateTime"]; + updatedAt?:PartialObjects["DateTime"] +} & (PartialObjects["BookingEngineSchema"] | PartialObjects["BookingSchema"] | PartialObjects["OrderEngineSchema"]), + ["ImageComponent"]: { + __typename?: "ImageComponent"; + title?:string, + fileType?:string, + large?:string, + medium?:string, + small?:string + }, + ["ImageInput"]: { + title?:string, + fileType?:string, + large:string, + medium?:string, + small?:string +}, + ["InvoiceInfo"]: { + __typename?: "InvoiceInfo"; + provider?:PartialObjects["InvoicingProvider"], + invoiceId?:string, + invoiceNumber?:number + }, + ["InvoicingProvider"]:InvoicingProvider, + /** The `JSONObject` scalar type represents JSON objects as specified by [ECMA-404](http://www.ecma-international.org/publications/files/ECMA-ST/ECMA-404.pdf). */ +["JSONObject"]:any, + ["LineItemSchema"]: { + __typename?: "LineItemSchema"; + _id?:string, + productRessource?:PartialObjects["ProductRessourceEnum"], + productId?:string, + title?:string, + price?:number, + salesPrice?:number, + quantity?:number, + vatClassId?:string, + parentId?:string, + addOns?:PartialObjects["AddOnComponent"][], + localeInfo?:PartialObjects["LocaleInfo"], + promoId?:string, + finalPrice?:number, + getProduct?:PartialObjects["ProductUnion"], + getLinePrice?:number, + getLineVatPrice?:number + }, + ["LinkEmailInput"]: { + email:string, + password:string, + idToken:string +}, + ["ListEnum"]:ListEnum, + ["ListModel"]: { + __typename?: "ListModel"; + _id?:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt?:PartialObjects["DateTime"], + updatedAt?:PartialObjects["DateTime"], + r?:string[], + w?:string[], + d?:string[], + title?:PartialObjects["TranslatableComponent"], + teaser?:PartialObjects["TranslatableComponent"], + cover?:PartialObjects["ImageComponent"], + thumbnail?:PartialObjects["ImageComponent"], + extraImages?:PartialObjects["ImageComponent"][], + content?:PartialObjects["TranslatableComponent"], + seo?:PartialObjects["SEOField"], + urls?:PartialObjects["TranslatableComponent"], + ressourceType?:PartialObjects["ListEnum"], + colorCode?:string + }, + ["LocaleInfo"]: { + __typename?: "LocaleInfo"; + countryCode?:string, + locale?:PartialObjects["AvailableTranslation"] + }, + ["LocComponent"]: { + __typename?: "LocComponent"; + type?:string, + coordinates?:number[] + }, + ["LoginInput"]: { + email:string, + password:string +}, + ["MainImage"]: { + main?:PartialObjects["ImageInput"], + one?:PartialObjects["ImageInput"], + two?:PartialObjects["ImageInput"], + three?:PartialObjects["ImageInput"], + four?:PartialObjects["ImageInput"] +}, + ["MainImageComponent"]: { + __typename?: "MainImageComponent"; + main?:PartialObjects["ImageComponent"], + one?:PartialObjects["ImageComponent"], + two?:PartialObjects["ImageComponent"], + three?:PartialObjects["ImageComponent"], + four?:PartialObjects["ImageComponent"] + }, + ["MessageModel"]: { + __typename?: "MessageModel"; + _id?:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt?:PartialObjects["DateTime"], + updatedAt?:PartialObjects["DateTime"], + r?:string[], + w?:string[], + d?:string[], + roomId?:string, + notified?:boolean, + message?:string, + edited?:boolean, + deleted?:boolean, + file?:PartialObjects["ImageComponent"], + sentBy?:string, + readBy?:PartialObjects["MessageReadBy"][] + }, + ["MessageReadBy"]: { + __typename?: "MessageReadBy"; + accountId?:string, + readAt?:PartialObjects["DateTime"] + }, + ["MetaBy"]: { + __typename?: "MetaBy"; + createdBy?:string, + updatedBy?:string, + deletedBy?:string + }, + ["MetaPermissions"]: { + __typename?: "MetaPermissions"; + r?:string[], + w?:string[], + d?:string[] + }, + ["ModelLoadersEnum"]:ModelLoadersEnum, + ["Mutation"]: { + __typename?: "Mutation"; + updateMe?:PartialObjects["AccountModel"], + updateMeEmail?:PartialObjects["AccountModel"], + updateMePassword?:PartialObjects["AccountModel"], + resetPassword?:PartialObjects["SimpleResult"], + registerGuest?:PartialObjects["AccountModel"], + register?:PartialObjects["AccountModel"], + accountsBillingInfosAddOne?:PartialObjects["BillingAddressComponent"], + accountsBillingInfosEditOne?:PartialObjects["BillingAddressComponent"], + accountsBillingInfosDeleteOne?:string, + accountsPaymentMethodsAddOne?:PartialObjects["PaymentMethodDetail"], + accountsPaymentMethodsMarkasDefault?:PartialObjects["PaymentMethodDetail"], + accountsPaymentMethodsDeleteOne?:string, + accountsLinkWithStripe?:string, + accountsRefreshInfoStripe?:PartialObjects["AccountModel"], + accountsFavoritesAddOne?:PartialObjects["FavoriteComponent"], + accountsFavoritesRemoveOne?:PartialObjects["SuccessResponse"], + workspacesAddOne?:PartialObjects["WorkspaceModel"], + workspacesEditOne?:PartialObjects["WorkspaceModel"], + workspacesDeleteOne?:PartialObjects["WorkspaceModel"], + finaliseCheckoutWithStripe?:PartialObjects["OrderEngineSchema"], + removeCurrentOrder?:PartialObjects["OrderEngineSchema"], + validateOrder?:PartialObjects["OrderEngineSchema"], + roomsCreateOne?:PartialObjects["RoomModel"], + roomsAddPeople?:PartialObjects["RoomModel"], + roomsRemovePeople?:PartialObjects["RoomModel"], + roomsMarkAllMessageAsRead?:PartialObjects["RoomModel"], + reserveOneBooking?:PartialObjects["OrderEngineSchema"], + roomsAddOneMessage?:PartialObjects["MessageModel"], + roomsEditOneMessage?:PartialObjects["MessageModel"], + roomsMarkMessageAsRead?:PartialObjects["MessageModel"], + roomsDeleteOneMessage?:PartialObjects["MessageModel"] + }, + ["NewAccountInput"]: { + email:string, + password:string, + passwordConfirmation:string, + userName?:string, + firstName?:string, + lastName?:string, + profilePicture?:PartialObjects["ImageInput"], + phoneNumber?:string, + gender?:PartialObjects["AccountGenderEnum"], + birthDate?:PartialObjects["DateTime"], + address?:PartialObjects["AddressInput"], + language?:PartialObjects["AvailableTranslation"], + settings?:PartialObjects["SettingsInput"][], + terms:boolean +}, + ["NewEmailInput"]: { + newEmail:string +}, + ["NewMessageInput"]: { + roomId:string, + message:string, + file?:PartialObjects["ImageInput"] +}, + ["NewPasswordInput"]: { + oldPassword:string, + newPassword:string, + newPasswordConfirmation:string +}, + ["NewRoomInput"]: { + accountIds:string[] +}, + ["NewWorkspaceInput"]: { + title?:string, + teaser?:string, + cover?:PartialObjects["ImageInput"], + thumbnail?:PartialObjects["ImageInput"], + extraImages?:PartialObjects["ImageInput"][], + content?:string, + seo?:PartialObjects["SEOSimpleInput"], + urls?:string, + workspaceType:PartialObjects["WorkspaceTypeEnum"], + pricingPerHour:number, + currency:PartialObjects["AvailableCurrency"], + maxCapacity:number, + minStay:number, + maxStay?:number, + availability:PartialObjects["AvailabilityInput"], + mainImages:PartialObjects["MainImage"], + equipmentIds:string[], + featureIds:string[], + address:PartialObjects["AddressStrictInput"] +}, + ["OrderBusinessStatusEnum"]:OrderBusinessStatusEnum, + ["OrderEngineSchema"]: { + __typename?: "OrderEngineSchema"; + lines?:PartialObjects["LineItemSchema"][], + localeInfo?:PartialObjects["LocaleInfo"], + currency?:PartialObjects["CurrencyEnum"], + vatExempt?:boolean, + accountId?:string, + contactInfo?:PartialObjects["BuyerInfoComponent"], + billingInfo?:PartialObjects["BuyerInfoComponent"], + promoId?:string, + invoiceInfo?:PartialObjects["InvoiceInfo"], + providerOrderItems?:PartialObjects["ProviderSchema"][], + orderName?:PartialObjects["TranslatableComponent"], + orderStatus?:PartialObjects["OrderStatusEnum"], + paymentIntent?:PartialObjects["PaymentIntentInfo"], + paymentStatus?:PartialObjects["PaymentStatusEnum"], + paymentInfo?:PartialObjects["PaymentInfo"], + subTotalPrice?:number, + vatPrice?:number, + finalPrice?:number, + promo?:PartialObjects["PromoModel"], + _id?:string, + organisationId?:string, + paths?:PartialObjects["EnginePathComponent"][], + by?:PartialObjects["MetaBy"], + permissions?:PartialObjects["MetaPermissions"], + createdAt?:PartialObjects["DateTime"], + updatedAt?:PartialObjects["DateTime"], + getBookingInput?:PartialObjects["BookingNewInput"] + }, + ["OrderStatusEnum"]:OrderStatusEnum, + ["PaymentInfo"]: { + __typename?: "PaymentInfo"; + provider?:PartialObjects["PaymentProvider"], + transactionId?:string + }, + ["PaymentIntentInfo"]: { + __typename?: "PaymentIntentInfo"; + provider?:PartialObjects["PaymentProvider"], + stripePaymentIntentData?:PartialObjects["StripePaymentIntentData"] + }, + ["PaymentMethodDetail"]: { + __typename?: "PaymentMethodDetail"; + _id?:string, + sPayMethodId?:string, + default?:boolean, + nameOnCard?:string, + type?:PartialObjects["cardTypeEnum"], + cardInfo?:PartialObjects["CardInfo"] + }, + ["PaymentMethodInput"]: { + sPayMethodId:string, + nameOnCard:string, + default?:boolean +}, + ["PaymentProvider"]:PaymentProvider, + ["PaymentStatusEnum"]:PaymentStatusEnum, + ["PlaceComponent"]: { + __typename?: "PlaceComponent"; + placeId?:string, + address?:PartialObjects["AddressComponent"], + loc?:PartialObjects["LocComponent"], + formattedAddress?:string + }, + ["ProductRessourceEnum"]:ProductRessourceEnum, + ["ProductUnion"]: PartialObjects["WorkspaceModel"], + ["PromoModel"]: { + __typename?: "PromoModel"; + _id?:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt?:PartialObjects["DateTime"], + updatedAt?:PartialObjects["DateTime"], + r?:string[], + w?:string[], + d?:string[], + code?:string, + description?:PartialObjects["TranslatableComponent"], + type?:PartialObjects["PromoType"], + value?:number, + validity?:PartialObjects["DateTime"], + cummulable?:boolean, + usageLimit?:number, + usage?:number + }, + ["PromoType"]:PromoType, + ["ProviderSchema"]: { + __typename?: "ProviderSchema"; + _id?:string, + organisationId?:string, + orderStatus?:PartialObjects["OrderBusinessStatusEnum"], + lines?:PartialObjects["LineItemSchema"][] + }, + ["Query"]: { + __typename?: "Query"; + me?:PartialObjects["AccountModel"], + login?:PartialObjects["FirebaseTokenResult"], + magicLink?:PartialObjects["SimpleResult"], + placesAutocomplete?:PartialObjects["JSONObject"][], + placesGeocode?:PartialObjects["JSONObject"], + placesGeocodeFromAddress?:PartialObjects["JSONObject"], + categoriesGetOne?:PartialObjects["CategoryModel"], + categoriesGetMany?:PartialObjects["CategoryModel"][], + categoriesGetCount?:number, + listsGetOne?:PartialObjects["ListModel"], + listsGetMany?:PartialObjects["ListModel"][], + listsGetCount?:number, + accountsDashboardLinkStripe?:string, + workspacesGetOne?:PartialObjects["WorkspaceModel"], + workspacesGetMany?:PartialObjects["WorkspaceModel"][], + workspacesGetCount?:number, + workspacesSearchMany?:PartialObjects["WorkspaceModel"][], + workspacesGetMine?:PartialObjects["WorkspaceModel"][], + workspacesGetMineCount?:number, + bookingsGetOne?:PartialObjects["BookingEngineSchema"], + bookingsGetMany?:PartialObjects["BookingEngineSchema"][], + bookingsGetCount?:number, + myCurrentOrder?:PartialObjects["OrderEngineSchema"], + myOrdersGetOne?:PartialObjects["OrderEngineSchema"], + myOrdersGetMany?:PartialObjects["OrderEngineSchema"][], + myOrdersGetManyCount?:number, + roomsGetOne?:PartialObjects["RoomModel"], + roomsGetMany?:PartialObjects["RoomModel"][], + roomsGetMessages?:PartialObjects["MessageModel"][] + }, + ["RangeType"]:RangeType, + ["RemoveInput"]: { + ressourceId:string +}, + ["RemovePeopleFromRoomInput"]: { + roomId:string, + accountIds:string[] +}, + ["ResetPasswordInput"]: { + email:string +}, + ["RessourceEnum"]:RessourceEnum, + ["RoomModel"]: { + __typename?: "RoomModel"; + _id?:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt?:PartialObjects["DateTime"], + updatedAt?:PartialObjects["DateTime"], + r?:string[], + w?:string[], + d?:string[], + title?:string, + stats?:PartialObjects["RoomsStats"], + getAccounts?:PartialObjects["AccountModel"][], + getMessages?:PartialObjects["MessageModel"][], + lastMessage?:PartialObjects["DateTime"] + }, + ["RoomsStats"]: { + __typename?: "RoomsStats"; + unreadCounts?:PartialObjects["UnreadCount"][] + }, + ["SEOField"]: { + __typename?: "SEOField"; + title?:PartialObjects["TranslatableComponent"], + description?:PartialObjects["TranslatableComponent"], + keywords?:PartialObjects["TranslatableComponent"], + thumbnail?:PartialObjects["ImageComponent"] + }, + ["SEOSimpleField"]: { + __typename?: "SEOSimpleField"; + title?:string, + description?:string, + keywords?:string, + thumbnail?:PartialObjects["ImageComponent"] + }, + ["SEOSimpleInput"]: { + title:string, + description:string, + keywords?:string, + thumbnail?:PartialObjects["ImageInput"] +}, + ["SettingsComponent"]: { + __typename?: "SettingsComponent"; + type?:PartialObjects["SettingsTypeEnum"], + email?:boolean, + pushNotifications?:boolean, + sms?:boolean + }, + ["SettingsInput"]: { + type:PartialObjects["SettingsTypeEnum"], + email?:boolean, + pushNotifications?:boolean, + sms?:boolean +}, + ["SettingsTypeEnum"]:SettingsTypeEnum, + ["SimpleResult"]: { + __typename?: "SimpleResult"; + message?:string + }, + ["SlotComponent"]: { + __typename?: "SlotComponent"; + startDate?:PartialObjects["DateTime"], + endDate?:PartialObjects["DateTime"], + startTime?:string, + endTime?:string + }, + ["SlotComponentInput"]: { + startDate:PartialObjects["DateTime"], + endDate:PartialObjects["DateTime"], + startTime:string, + endTime:string +}, + ["StatusEnum"]:StatusEnum, + ["StripeInfo"]: { + __typename?: "StripeInfo"; + customerId?:string + }, + ["StripePaymentIntentData"]: { + __typename?: "StripePaymentIntentData"; + id?:string, + client_secret?:string, + currency?:string, + customer?:string, + status?:string + }, + ["StripePayoutInfo"]: { + __typename?: "StripePayoutInfo"; + accountId?:string, + chargesEnabled?:boolean, + payoutsEnabled?:boolean, + detailsSubmitted?:boolean + }, + ["SuccessResponse"]: { + __typename?: "SuccessResponse"; + success?:boolean + }, + ["TranslatableComponent"]: { + __typename?: "TranslatableComponent"; + en?:string, + fr?:string, + nl?:string, + de?:string + }, + ["UnreadCount"]: { + __typename?: "UnreadCount"; + accountId?:string, + count?:number + }, + ["WorkspaceModel"]: { + __typename?: "WorkspaceModel"; + _id?:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt?:PartialObjects["DateTime"], + updatedAt?:PartialObjects["DateTime"], + r?:string[], + w?:string[], + d?:string[], + title?:string, + teaser?:string, + cover?:PartialObjects["ImageComponent"], + thumbnail?:PartialObjects["ImageComponent"], + extraImages?:PartialObjects["ImageComponent"][], + content?:string, + seo?:PartialObjects["SEOSimpleField"], + urls?:string, + place?:PartialObjects["PlaceComponent"], + dist?:PartialObjects["GeolocDistComponent"], + getDistFromLocation?:number, + workspaceType?:PartialObjects["WorkspaceTypeEnum"], + pricingPerHour?:number, + currency?:PartialObjects["AvailableCurrency"], + maxCapacity?:number, + minStay?:number, + maxStay?:number, + availability?:PartialObjects["AvailabilityComponent"], + mainImages?:PartialObjects["MainImageComponent"], + equipmentIds?:string[], + featureIds?:string[], + address?:PartialObjects["AddressComponent"], + finalPrice?:number, + getEquipments?:(PartialObjects["ListModel"] | undefined)[], + getFeatures?:(PartialObjects["ListModel"] | undefined)[], + getBookings?:(PartialObjects["BookingSchema"] | undefined)[], + getOwner?:PartialObjects["WorkspaceOwnerComponent"] + }, + ["WorkspaceOwnerComponent"]: { + __typename?: "WorkspaceOwnerComponent"; + firstName?:string, + lastName?:string + }, + ["WorkspaceTypeEnum"]:WorkspaceTypeEnum + } + +export enum AccountGenderEnum { + m = "m", + f = "f", + other = "other" +} + +export type AccountModel = { + __typename?: "AccountModel", + _id:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt:DateTime, + updatedAt:DateTime, + r:string[], + w:string[], + d:string[], + types:AccountTypeEnum[], + organisationIds?:string[], + email:string, + userName?:string, + firstName?:string, + lastName?:string, + profilePicture?:ImageComponent, + phoneNumber?:string, + gender?:AccountGenderEnum, + birthDate?:DateTime, + getAge?:number, + address?:AddressComponent, + language?:AvailableTranslation, + settings?:SettingsComponent[], + paymentInfo?:AccountPaymentInfo, + payoutInfo?:AccountPayoutInfo, + terms:boolean, + favLikes?:(FavoriteComponent | undefined)[], + getFavorites:FavoriteComponent[], + getLiked:FavoriteComponent[] +} + +export type AccountPaymentInfo = { + __typename?: "AccountPaymentInfo", + stripeInfo?:StripeInfo, + paymentMethods?:PaymentMethodDetail[], + billingInfos?:BillingAddressComponent[] +} + +export type AccountPayoutInfo = { + __typename?: "AccountPayoutInfo", + stripeInfo?:StripePayoutInfo +} + +export enum AccountTypeEnum { + admin = "admin", + user = "user", + pro = "pro", + public = "public", + organisationOwner = "organisationOwner", + storeManager = "storeManager" +} + +export type AddInput = { + ressourceId:string, + ressourceType:RessourceEnum +} + +export type AddOnComponent = { + __typename?: "AddOnComponent", + addOnId:string, + quantity:number +} + +export type AddPeopleToRoomInput = { + roomId:string, + accountIds:string[] +} + +export type AddressComponent = { + __typename?: "AddressComponent", + number?:string, + street?:string, + streetBis?:string, + floor?:string, + box?:string, + zip?:string, + state?:string, + city:string, + country:string +} + +export type AddressInput = { + number?:string, + street?:string, + streetBis?:string, + floor?:string, + box?:string, + zip?:string, + state?:string, + city:string, + country:string +} + +export type AddressStrictComponent = { + __typename?: "AddressStrictComponent", + number:string, + street:string, + zip:string, + city:string, + country:string, + streetBis?:string, + floor?:string, + box?:string, + state?:string +} + +export type AddressStrictInput = { + number:string, + street:string, + zip:string, + city:string, + country:string, + streetBis?:string, + floor?:string, + box?:string, + state?:string +} + +export type AvailabilityComponent = { + __typename?: "AvailabilityComponent", + dates:DateRangeComponent, + hours?:HourComponent[], + exceptions?:DateExceptionComponent[], + noWeekend?:boolean +} + +export type AvailabilityInput = { + dates:DateRangeComponentInput, + hours?:HoursInput[], + exceptions?:DateExceptionComponentInput[], + noWeekend?:boolean +} + +export enum AvailableCurrency { + eur = "eur", + usd = "usd" +} + +export enum AvailableTranslation { + en = "en", + fr = "fr", + nl = "nl", + de = "de" +} + +export type BillingAddressComponent = { + __typename?: "BillingAddressComponent", + firstName:string, + lastName:string, + address:AddressStrictComponent, + email?:string, + default?:boolean, + customTags?:JSONObject, + company?:string, + vatNumber?:string, + fiscalForm?:string, + _id:string +} + +export type BillingAddressInput = { + firstName:string, + lastName:string, + address:AddressStrictInput, + email?:string, + default?:boolean, + customTags?:JSONObject, + company?:string, + vatNumber?:string, + fiscalForm?:string +} + +export type BookingEngineSchema = { + __typename?: "BookingEngineSchema", + durationType:DurationTypeEnum, + startToEnd?:DateRangeComponent, + slot?:SlotComponent, + capacity?:number, + comments?:string, + status:StatusEnum, + paths?:EnginePathComponent[], + ownerId:string, + dates:DateRangeComponent[], + owner:AccountModel, + _id:string, + organisationId?:string, + by?:MetaBy, + permissions?:MetaPermissions, + createdAt:DateTime, + updatedAt:DateTime, + workspace?:WorkspaceModel, + room?:RoomModel +} + +export type BookingNewInput = { + __typename?: "BookingNewInput", + durationType:DurationTypeEnum, + startToEnd?:DateRangeComponent, + slot?:SlotComponent, + capacity?:number, + comments?:string, + ressourceModel:BookingsRessourceEnum, + ressourceId:string, + checkoutInfo?:CheckoutEngineInput +} + +export type BookingNewInputSchema = { + durationType:DurationTypeEnum, + startToEnd?:DateRangeComponentInput, + slot?:SlotComponentInput, + capacity?:number, + comments?:string, + ressourceModel:BookingsRessourceEnum, + ressourceId:string, + checkoutInfo?:CheckoutEngineInputI +} + +export type BookingSchema = { + __typename?: "BookingSchema", + durationType:DurationTypeEnum, + startToEnd?:DateRangeComponent, + slot?:SlotComponent, + capacity:number, + comments?:string, + status:StatusEnum, + paths?:EnginePathComponent[], + ownerId:string, + dates:DateRangeComponent[], + _id:string, + organisationId?:string, + by?:MetaBy, + permissions?:MetaPermissions, + createdAt:DateTime, + updatedAt:DateTime +} + +export enum BookingsRessourceEnum { + workspaces = "workspaces" +} + +export type BuyerInfo = { + firstName:string, + lastName:string, + email:string, + company?:string, + vatNumber?:string, + address:AddressInput +} + +export type BuyerInfoComponent = { + __typename?: "BuyerInfoComponent", + firstName:string, + lastName:string, + email:string, + company?:string, + vatNumber?:string, + address:AddressComponent +} + +export enum cardBrandEnum { + amex = "amex", + diners = "diners", + discover = "discover", + jcb = "jcb", + mastercard = "mastercard", + unionpay = "unionpay", + visa = "visa", + unknown = "unknown" +} + +export type CardInfo = { + __typename?: "CardInfo", + brand:cardBrandEnum, + country:string, + exp_month:number, + exp_year:number, + last4:string +} + +export enum cardTypeEnum { + alipay = "alipay", + au_becs_debit = "au_becs_debit", + bacs_debit = "bacs_debit", + bancontact = "bancontact", + card = "card", + eps = "eps", + fpx = "fpx", + giropay = "giropay", + ideal = "ideal", + p24 = "p24", + sepa_debit = "sepa_debit", + sofort = "sofort" +} + +export type CategoryModel = { + __typename?: "CategoryModel", + _id:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt:DateTime, + updatedAt:DateTime, + r:string[], + w:string[], + d:string[], + title:TranslatableComponent, + teaser?:TranslatableComponent, + cover?:ImageComponent, + thumbnail?:ImageComponent, + extraImages?:ImageComponent[], + content?:TranslatableComponent, + seo?:SEOField, + urls?:TranslatableComponent, + ressourceType:RessourceEnum, + colorCode?:string +} + +export type CheckoutEngineInput = { + __typename?: "CheckoutEngineInput", + contactInfo:BuyerInfoComponent, + billingInfo:BuyerInfoComponent, + token?:string +} + +export type CheckoutEngineInputI = { + contactInfo:BuyerInfo, + billingInfo:BuyerInfo, + token?:string +} + +export type CheckoutInput = { + contactInfo:BuyerInfo, + billingInfo:BuyerInfo, + token?:string +} + +export enum CountryCodesComponentEnum { + AF = "AF", + AX = "AX", + AL = "AL", + DZ = "DZ", + AS = "AS", + AD = "AD", + AO = "AO", + AI = "AI", + AQ = "AQ", + AG = "AG", + AR = "AR", + AM = "AM", + AW = "AW", + AU = "AU", + AT = "AT", + AZ = "AZ", + BS = "BS", + BH = "BH", + BD = "BD", + BB = "BB", + BY = "BY", + BE = "BE", + BZ = "BZ", + BJ = "BJ", + BM = "BM", + BT = "BT", + BO = "BO", + BA = "BA", + BW = "BW", + BV = "BV", + BR = "BR", + IO = "IO", + BN = "BN", + BG = "BG", + BF = "BF", + BI = "BI", + KH = "KH", + CM = "CM", + CA = "CA", + CV = "CV", + KY = "KY", + CF = "CF", + TD = "TD", + CL = "CL", + CN = "CN", + CX = "CX", + CC = "CC", + CO = "CO", + KM = "KM", + CG = "CG", + CD = "CD", + CK = "CK", + CR = "CR", + CI = "CI", + HR = "HR", + CU = "CU", + CY = "CY", + CZ = "CZ", + DK = "DK", + DJ = "DJ", + DM = "DM", + DO = "DO", + EC = "EC", + EG = "EG", + SV = "SV", + GQ = "GQ", + ER = "ER", + EE = "EE", + ET = "ET", + FK = "FK", + FO = "FO", + FJ = "FJ", + FI = "FI", + FR = "FR", + GF = "GF", + PF = "PF", + TF = "TF", + GA = "GA", + GM = "GM", + GE = "GE", + DE = "DE", + GH = "GH", + GI = "GI", + GR = "GR", + GL = "GL", + GD = "GD", + GP = "GP", + GU = "GU", + GT = "GT", + GG = "GG", + GN = "GN", + GW = "GW", + GY = "GY", + HT = "HT", + HM = "HM", + VA = "VA", + HN = "HN", + HK = "HK", + HU = "HU", + IS = "IS", + IN = "IN", + ID = "ID", + IR = "IR", + IQ = "IQ", + IE = "IE", + IM = "IM", + IL = "IL", + IT = "IT", + JM = "JM", + JP = "JP", + JE = "JE", + JO = "JO", + KZ = "KZ", + KE = "KE", + KI = "KI", + KR = "KR", + KW = "KW", + KG = "KG", + LA = "LA", + LV = "LV", + LB = "LB", + LS = "LS", + LR = "LR", + LY = "LY", + LI = "LI", + LT = "LT", + LU = "LU", + MO = "MO", + MK = "MK", + MG = "MG", + MW = "MW", + MY = "MY", + MV = "MV", + ML = "ML", + MT = "MT", + MH = "MH", + MQ = "MQ", + MR = "MR", + MU = "MU", + YT = "YT", + MX = "MX", + FM = "FM", + MD = "MD", + MC = "MC", + MN = "MN", + ME = "ME", + MS = "MS", + MA = "MA", + MZ = "MZ", + MM = "MM", + NA = "NA", + NR = "NR", + NP = "NP", + NL = "NL", + AN = "AN", + NC = "NC", + NZ = "NZ", + NI = "NI", + NE = "NE", + NG = "NG", + NU = "NU", + NF = "NF", + MP = "MP", + NO = "NO", + OM = "OM", + PK = "PK", + PW = "PW", + PS = "PS", + PA = "PA", + PG = "PG", + PY = "PY", + PE = "PE", + PH = "PH", + PN = "PN", + PL = "PL", + PT = "PT", + PR = "PR", + QA = "QA", + RE = "RE", + RO = "RO", + RU = "RU", + RW = "RW", + BL = "BL", + SH = "SH", + KN = "KN", + LC = "LC", + MF = "MF", + PM = "PM", + VC = "VC", + WS = "WS", + SM = "SM", + ST = "ST", + SA = "SA", + SN = "SN", + RS = "RS", + SC = "SC", + SL = "SL", + SG = "SG", + SK = "SK", + SI = "SI", + SB = "SB", + SO = "SO", + ZA = "ZA", + GS = "GS", + ES = "ES", + LK = "LK", + SD = "SD", + SR = "SR", + SJ = "SJ", + SZ = "SZ", + SE = "SE", + CH = "CH", + SY = "SY", + TW = "TW", + TJ = "TJ", + TZ = "TZ", + TH = "TH", + TL = "TL", + TG = "TG", + TK = "TK", + TO = "TO", + TT = "TT", + TN = "TN", + TR = "TR", + TM = "TM", + TC = "TC", + TV = "TV", + UG = "UG", + UA = "UA", + AE = "AE", + GB = "GB", + US = "US", + UM = "UM", + UY = "UY", + UZ = "UZ", + VU = "VU", + VE = "VE", + VN = "VN", + VG = "VG", + VI = "VI", + WF = "WF", + EH = "EH", + YE = "YE", + ZM = "ZM", + ZW = "ZW", + XK = "XK", + KP = "KP" +} + +export enum CurrencyEnum { + eur = "eur", + usd = "usd" +} + +export type DateExceptionComponent = { + __typename?: "DateExceptionComponent", + dates:DateRangeComponent, + hours?:HourComponent[], + allDay?:boolean +} + +export type DateExceptionComponentInput = { + dates:DateRangeComponentInput, + hours?:HoursInput[], + allDay?:boolean +} + +export type DateRangeComponent = { + __typename?: "DateRangeComponent", + startDate:DateTime, + endDate:DateTime +} + +export type DateRangeComponentInput = { + startDate:DateTime, + endDate:DateTime +} + +export enum DateTabType { + today = "today", + tomorrow = "tomorrow", + upcomming = "upcomming", + past = "past" +} + +/** The javascript `Date` as string. Type represents date and time as the ISO Date string. */ +export type DateTime = any + +export enum DurationTypeEnum { + startToEnd = "startToEnd", + slot = "slot" +} + +export type EditAccountInput = { + userName?:string, + firstName?:string, + lastName?:string, + profilePicture?:ImageInput, + phoneNumber?:string, + gender?:AccountGenderEnum, + birthDate?:DateTime, + address?:AddressInput, + language?:AvailableTranslation, + settings?:SettingsInput[] +} + +export type EditMessageInput = { + message:string +} + +export type EditWorkspaceInput = { + title?:string, + teaser?:string, + cover?:ImageInput, + thumbnail?:ImageInput, + extraImages?:ImageInput[], + content?:string, + seo?:SEOSimpleInput, + urls?:string, + workspaceType?:WorkspaceTypeEnum, + pricingPerHour?:number, + currency:AvailableCurrency, + maxCapacity?:number, + minStay?:number, + maxStay?:number, + availability?:AvailabilityInput, + mainImages:MainImage, + equipmentIds?:string[], + featureIds?:string[], + address?:AddressStrictInput +} + +export type EnginePathComponent = { + __typename?: "EnginePathComponent", + ressourceModel:ModelLoadersEnum, + ressourceId:string +} + +export type FavoriteComponent = { + __typename?: "FavoriteComponent", + ressourceId:string, + ressourceType:RessourceEnum, + addedAt:DateTime, + favorite?:boolean, + liked?:boolean +} + +export type FirebaseTokenResult = { + __typename?: "FirebaseTokenResult", + localId:string, + email?:string, + displayName?:string, + idToken:string, + registered?:boolean, + refreshToken:string, + expiresIn:string +} + +export type GeolocAddressSearchInput = { + formattedAddress:string, + radius?:number +} + +export type GeolocDistComponent = { + __typename?: "GeolocDistComponent", + calculated?:number, + location?:LocComponent +} + +export type GeolocSearchInput = { + longitude:number, + latitude:number, + radius?:number +} + +export type GetArgs = { + limit:number, + skip:number, + sort?:string +} + +export type HourComponent = { + __typename?: "HourComponent", + from:string, + to:string +} + +export type HoursInput = { + from:string, + to:string +} + +export type IEngineSchema = { + __interface:{ + _id:string, + organisationId?:string, + paths?:EnginePathComponent[], + by?:MetaBy, + permissions?:MetaPermissions, + createdAt:DateTime, + updatedAt:DateTime + }; + __resolve:{ + ['...on BookingEngineSchema']: BookingEngineSchema; + ['...on BookingSchema']: BookingSchema; + ['...on OrderEngineSchema']: OrderEngineSchema; + } +} + +export type ImageComponent = { + __typename?: "ImageComponent", + title?:string, + fileType?:string, + large:string, + medium?:string, + small?:string +} + +export type ImageInput = { + title?:string, + fileType?:string, + large:string, + medium?:string, + small?:string +} + +export type InvoiceInfo = { + __typename?: "InvoiceInfo", + provider:InvoicingProvider, + invoiceId:string, + invoiceNumber:number +} + +export enum InvoicingProvider { + directInvoice = "directInvoice" +} + +/** The `JSONObject` scalar type represents JSON objects as specified by [ECMA-404](http://www.ecma-international.org/publications/files/ECMA-ST/ECMA-404.pdf). */ +export type JSONObject = any + +export type LineItemSchema = { + __typename?: "LineItemSchema", + _id:string, + productRessource:ProductRessourceEnum, + productId:string, + title:string, + price:number, + salesPrice:number, + quantity:number, + vatClassId:string, + parentId?:string, + addOns?:AddOnComponent[], + localeInfo?:LocaleInfo, + promoId?:string, + finalPrice?:number, + getProduct?:ProductUnion, + getLinePrice:number, + getLineVatPrice:number +} + +export type LinkEmailInput = { + email:string, + password:string, + idToken:string +} + +export enum ListEnum { + equipment = "equipment", + feature = "feature" +} + +export type ListModel = { + __typename?: "ListModel", + _id:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt:DateTime, + updatedAt:DateTime, + r:string[], + w:string[], + d:string[], + title:TranslatableComponent, + teaser?:TranslatableComponent, + cover?:ImageComponent, + thumbnail?:ImageComponent, + extraImages?:ImageComponent[], + content?:TranslatableComponent, + seo?:SEOField, + urls?:TranslatableComponent, + ressourceType:ListEnum, + colorCode?:string +} + +export type LocaleInfo = { + __typename?: "LocaleInfo", + countryCode?:string, + locale?:AvailableTranslation +} + +export type LocComponent = { + __typename?: "LocComponent", + type:string, + coordinates:number[] +} + +export type LoginInput = { + email:string, + password:string +} + +export type MainImage = { + main?:ImageInput, + one?:ImageInput, + two?:ImageInput, + three?:ImageInput, + four?:ImageInput +} + +export type MainImageComponent = { + __typename?: "MainImageComponent", + main?:ImageComponent, + one?:ImageComponent, + two?:ImageComponent, + three?:ImageComponent, + four?:ImageComponent +} + +export type MessageModel = { + __typename?: "MessageModel", + _id:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt:DateTime, + updatedAt:DateTime, + r:string[], + w:string[], + d:string[], + roomId:string, + notified?:boolean, + message:string, + edited:boolean, + deleted:boolean, + file?:ImageComponent, + sentBy:string, + readBy:MessageReadBy[] +} + +export type MessageReadBy = { + __typename?: "MessageReadBy", + accountId:string, + readAt:DateTime +} + +export type MetaBy = { + __typename?: "MetaBy", + createdBy?:string, + updatedBy?:string, + deletedBy?:string +} + +export type MetaPermissions = { + __typename?: "MetaPermissions", + r:string[], + w:string[], + d:string[] +} + +export enum ModelLoadersEnum { + streams = "streams", + accounts = "accounts", + workspaces = "workspaces", + bookings = "bookings" +} + +export type Mutation = { + __typename?: "Mutation", + updateMe:AccountModel, + updateMeEmail:AccountModel, + updateMePassword:AccountModel, + resetPassword:SimpleResult, + registerGuest:AccountModel, + register:AccountModel, + accountsBillingInfosAddOne:BillingAddressComponent, + accountsBillingInfosEditOne:BillingAddressComponent, + accountsBillingInfosDeleteOne:string, + accountsPaymentMethodsAddOne:PaymentMethodDetail, + accountsPaymentMethodsMarkasDefault:PaymentMethodDetail, + accountsPaymentMethodsDeleteOne:string, + accountsLinkWithStripe:string, + accountsRefreshInfoStripe:AccountModel, + accountsFavoritesAddOne:FavoriteComponent, + accountsFavoritesRemoveOne:SuccessResponse, + workspacesAddOne:WorkspaceModel, + workspacesEditOne:WorkspaceModel, + workspacesDeleteOne:WorkspaceModel, + finaliseCheckoutWithStripe:OrderEngineSchema, + removeCurrentOrder:OrderEngineSchema, + validateOrder:OrderEngineSchema, + roomsCreateOne:RoomModel, + roomsAddPeople:RoomModel, + roomsRemovePeople:RoomModel, + roomsMarkAllMessageAsRead:RoomModel, + reserveOneBooking:OrderEngineSchema, + roomsAddOneMessage:MessageModel, + roomsEditOneMessage:MessageModel, + roomsMarkMessageAsRead:MessageModel, + roomsDeleteOneMessage:MessageModel +} + +export type NewAccountInput = { + email:string, + password:string, + passwordConfirmation:string, + userName?:string, + firstName?:string, + lastName?:string, + profilePicture?:ImageInput, + phoneNumber?:string, + gender?:AccountGenderEnum, + birthDate?:DateTime, + address?:AddressInput, + language?:AvailableTranslation, + settings?:SettingsInput[], + terms:boolean +} + +export type NewEmailInput = { + newEmail:string +} + +export type NewMessageInput = { + roomId:string, + message:string, + file?:ImageInput +} + +export type NewPasswordInput = { + oldPassword:string, + newPassword:string, + newPasswordConfirmation:string +} + +export type NewRoomInput = { + accountIds:string[] +} + +export type NewWorkspaceInput = { + title?:string, + teaser?:string, + cover?:ImageInput, + thumbnail?:ImageInput, + extraImages?:ImageInput[], + content?:string, + seo?:SEOSimpleInput, + urls?:string, + workspaceType:WorkspaceTypeEnum, + pricingPerHour:number, + currency:AvailableCurrency, + maxCapacity:number, + minStay:number, + maxStay?:number, + availability:AvailabilityInput, + mainImages:MainImage, + equipmentIds:string[], + featureIds:string[], + address:AddressStrictInput +} + +export enum OrderBusinessStatusEnum { + processing = "processing", + shipped = "shipped", + completed = "completed", + canceled = "canceled", + refund = "refund" +} + +export type OrderEngineSchema = { + __typename?: "OrderEngineSchema", + lines:LineItemSchema[], + localeInfo:LocaleInfo, + currency:CurrencyEnum, + vatExempt:boolean, + accountId:string, + contactInfo?:BuyerInfoComponent, + billingInfo?:BuyerInfoComponent, + promoId?:string, + invoiceInfo?:InvoiceInfo, + providerOrderItems?:ProviderSchema[], + orderName?:TranslatableComponent, + orderStatus:OrderStatusEnum, + paymentIntent:PaymentIntentInfo, + paymentStatus?:PaymentStatusEnum, + paymentInfo?:PaymentInfo, + subTotalPrice:number, + vatPrice:number, + finalPrice:number, + promo?:PromoModel, + _id:string, + organisationId?:string, + paths?:EnginePathComponent[], + by?:MetaBy, + permissions?:MetaPermissions, + createdAt:DateTime, + updatedAt:DateTime, + getBookingInput?:BookingNewInput +} + +export enum OrderStatusEnum { + draft = "draft", + processing = "processing", + shipped = "shipped", + completed = "completed", + canceled = "canceled", + refund = "refund" +} + +export type PaymentInfo = { + __typename?: "PaymentInfo", + provider:PaymentProvider, + transactionId:string +} + +export type PaymentIntentInfo = { + __typename?: "PaymentIntentInfo", + provider:PaymentProvider, + stripePaymentIntentData:StripePaymentIntentData +} + +export type PaymentMethodDetail = { + __typename?: "PaymentMethodDetail", + _id:string, + sPayMethodId:string, + default:boolean, + nameOnCard:string, + type:cardTypeEnum, + cardInfo?:CardInfo +} + +export type PaymentMethodInput = { + sPayMethodId:string, + nameOnCard:string, + default?:boolean +} + +export enum PaymentProvider { + free = "free", + stripe = "stripe", + bancontact = "bancontact", + bankWire = "bankWire" +} + +export enum PaymentStatusEnum { + pending = "pending", + actionNeeded = "actionNeeded", + paid = "paid", + free = "free" +} + +export type PlaceComponent = { + __typename?: "PlaceComponent", + placeId:string, + address?:AddressComponent, + loc:LocComponent, + formattedAddress:string +} + +export enum ProductRessourceEnum { + workspaces = "workspaces" +} + +export type ProductUnion = { + __union:WorkspaceModel; + __resolve:{ + ['...on WorkspaceModel']: WorkspaceModel; + } +} + +export type PromoModel = { + __typename?: "PromoModel", + _id:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt:DateTime, + updatedAt:DateTime, + r:string[], + w:string[], + d:string[], + code:string, + description?:TranslatableComponent, + type:PromoType, + value:number, + validity?:DateTime, + cummulable:boolean, + usageLimit?:number, + usage:number +} + +export enum PromoType { + fixed = "fixed", + percentage = "percentage" +} + +export type ProviderSchema = { + __typename?: "ProviderSchema", + _id:string, + organisationId:string, + orderStatus:OrderBusinessStatusEnum, + lines:LineItemSchema[] +} + +export type Query = { + __typename?: "Query", + me:AccountModel, + login:FirebaseTokenResult, + magicLink:SimpleResult, + placesAutocomplete:JSONObject[], + placesGeocode:JSONObject, + placesGeocodeFromAddress:JSONObject, + categoriesGetOne:CategoryModel, + categoriesGetMany:CategoryModel[], + categoriesGetCount:number, + listsGetOne:ListModel, + listsGetMany:ListModel[], + listsGetCount:number, + accountsDashboardLinkStripe:string, + workspacesGetOne:WorkspaceModel, + workspacesGetMany:WorkspaceModel[], + workspacesGetCount:number, + workspacesSearchMany:WorkspaceModel[], + workspacesGetMine:WorkspaceModel[], + workspacesGetMineCount:number, + bookingsGetOne:BookingEngineSchema, + bookingsGetMany:BookingEngineSchema[], + bookingsGetCount:number, + myCurrentOrder:OrderEngineSchema, + myOrdersGetOne:OrderEngineSchema, + myOrdersGetMany:OrderEngineSchema[], + myOrdersGetManyCount:number, + roomsGetOne:RoomModel, + roomsGetMany:RoomModel[], + roomsGetMessages:MessageModel[] +} + +export enum RangeType { + strict = "strict", + intersect = "intersect", + intersectLarge = "intersectLarge", + included = "included" +} + +export type RemoveInput = { + ressourceId:string +} + +export type RemovePeopleFromRoomInput = { + roomId:string, + accountIds:string[] +} + +export type ResetPasswordInput = { + email:string +} + +export enum RessourceEnum { + workspaces = "workspaces" +} + +export type RoomModel = { + __typename?: "RoomModel", + _id:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt:DateTime, + updatedAt:DateTime, + r:string[], + w:string[], + d:string[], + title?:string, + stats:RoomsStats, + getAccounts?:AccountModel[], + getMessages?:MessageModel[], + lastMessage?:DateTime +} + +export type RoomsStats = { + __typename?: "RoomsStats", + unreadCounts:UnreadCount[] +} + +export type SEOField = { + __typename?: "SEOField", + title:TranslatableComponent, + description:TranslatableComponent, + keywords?:TranslatableComponent, + thumbnail?:ImageComponent +} + +export type SEOSimpleField = { + __typename?: "SEOSimpleField", + title:string, + description:string, + keywords?:string, + thumbnail?:ImageComponent +} + +export type SEOSimpleInput = { + title:string, + description:string, + keywords?:string, + thumbnail?:ImageInput +} + +export type SettingsComponent = { + __typename?: "SettingsComponent", + type:SettingsTypeEnum, + email?:boolean, + pushNotifications?:boolean, + sms?:boolean +} + +export type SettingsInput = { + type:SettingsTypeEnum, + email?:boolean, + pushNotifications?:boolean, + sms?:boolean +} + +export enum SettingsTypeEnum { + notifications = "notifications", + promotions = "promotions", + reminders = "reminders" +} + +export type SimpleResult = { + __typename?: "SimpleResult", + message:string +} + +export type SlotComponent = { + __typename?: "SlotComponent", + startDate:DateTime, + endDate:DateTime, + startTime:string, + endTime:string +} + +export type SlotComponentInput = { + startDate:DateTime, + endDate:DateTime, + startTime:string, + endTime:string +} + +export enum StatusEnum { + draft = "draft", + processing = "processing", + validated = "validated", + cancelled = "cancelled" +} + +export type StripeInfo = { + __typename?: "StripeInfo", + customerId:string +} + +export type StripePaymentIntentData = { + __typename?: "StripePaymentIntentData", + id:string, + client_secret?:string, + currency:string, + customer:string, + status:string +} + +export type StripePayoutInfo = { + __typename?: "StripePayoutInfo", + accountId:string, + chargesEnabled:boolean, + payoutsEnabled:boolean, + detailsSubmitted:boolean +} + +export type SuccessResponse = { + __typename?: "SuccessResponse", + success:boolean +} + +export type TranslatableComponent = { + __typename?: "TranslatableComponent", + en?:string, + fr:string, + nl?:string, + de?:string +} + +export type UnreadCount = { + __typename?: "UnreadCount", + accountId:string, + count:number +} + +export type WorkspaceModel = { + __typename?: "WorkspaceModel", + _id:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt:DateTime, + updatedAt:DateTime, + r:string[], + w:string[], + d:string[], + title?:string, + teaser?:string, + cover?:ImageComponent, + thumbnail?:ImageComponent, + extraImages?:ImageComponent[], + content?:string, + seo?:SEOSimpleField, + urls?:string, + place:PlaceComponent, + dist?:GeolocDistComponent, + getDistFromLocation?:number, + workspaceType:WorkspaceTypeEnum, + pricingPerHour:number, + currency:AvailableCurrency, + maxCapacity:number, + minStay:number, + maxStay?:number, + availability:AvailabilityComponent, + mainImages:MainImageComponent, + equipmentIds:string[], + featureIds:string[], + address:AddressComponent, + finalPrice:number, + getEquipments?:(ListModel | undefined)[], + getFeatures?:(ListModel | undefined)[], + getBookings?:(BookingSchema | undefined)[], + getOwner?:WorkspaceOwnerComponent +} + +export type WorkspaceOwnerComponent = { + __typename?: "WorkspaceOwnerComponent", + firstName?:string, + lastName?:string +} + +export enum WorkspaceTypeEnum { + meeting = "meeting", + individual = "individual", + open = "open" +} + +export const AllTypesProps: Record = { + AccountGenderEnum: "enum", + AccountTypeEnum: "enum", + AddInput:{ + ressourceId:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + ressourceType:{ + type:"RessourceEnum", + array:false, + arrayRequired:false, + required:true + } + }, + AddPeopleToRoomInput:{ + roomId:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + accountIds:{ + type:"String", + array:true, + arrayRequired:true, + required:true + } + }, + AddressInput:{ + number:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + street:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + streetBis:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + floor:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + box:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + zip:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + state:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + city:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + country:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + AddressStrictInput:{ + number:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + street:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + zip:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + city:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + country:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + streetBis:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + floor:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + box:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + state:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + AvailabilityInput:{ + dates:{ + type:"DateRangeComponentInput", + array:false, + arrayRequired:false, + required:true + }, + hours:{ + type:"HoursInput", + array:true, + arrayRequired:false, + required:true + }, + exceptions:{ + type:"DateExceptionComponentInput", + array:true, + arrayRequired:false, + required:true + }, + noWeekend:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + } + }, + AvailableCurrency: "enum", + AvailableTranslation: "enum", + BillingAddressInput:{ + firstName:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + lastName:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + address:{ + type:"AddressStrictInput", + array:false, + arrayRequired:false, + required:true + }, + email:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + default:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + }, + customTags:{ + type:"JSONObject", + array:false, + arrayRequired:false, + required:false + }, + company:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + vatNumber:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + fiscalForm:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + BookingNewInputSchema:{ + durationType:{ + type:"DurationTypeEnum", + array:false, + arrayRequired:false, + required:true + }, + startToEnd:{ + type:"DateRangeComponentInput", + array:false, + arrayRequired:false, + required:false + }, + slot:{ + type:"SlotComponentInput", + array:false, + arrayRequired:false, + required:false + }, + capacity:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + comments:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + ressourceModel:{ + type:"BookingsRessourceEnum", + array:false, + arrayRequired:false, + required:true + }, + ressourceId:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + checkoutInfo:{ + type:"CheckoutEngineInputI", + array:false, + arrayRequired:false, + required:false + } + }, + BookingsRessourceEnum: "enum", + BuyerInfo:{ + firstName:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + lastName:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + email:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + company:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + vatNumber:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + address:{ + type:"AddressInput", + array:false, + arrayRequired:false, + required:true + } + }, + cardBrandEnum: "enum", + cardTypeEnum: "enum", + CheckoutEngineInputI:{ + contactInfo:{ + type:"BuyerInfo", + array:false, + arrayRequired:false, + required:true + }, + billingInfo:{ + type:"BuyerInfo", + array:false, + arrayRequired:false, + required:true + }, + token:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + CheckoutInput:{ + contactInfo:{ + type:"BuyerInfo", + array:false, + arrayRequired:false, + required:true + }, + billingInfo:{ + type:"BuyerInfo", + array:false, + arrayRequired:false, + required:true + }, + token:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + CountryCodesComponentEnum: "enum", + CurrencyEnum: "enum", + DateExceptionComponentInput:{ + dates:{ + type:"DateRangeComponentInput", + array:false, + arrayRequired:false, + required:true + }, + hours:{ + type:"HoursInput", + array:true, + arrayRequired:false, + required:true + }, + allDay:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + } + }, + DateRangeComponentInput:{ + startDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:true + }, + endDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:true + } + }, + DateTabType: "enum", + DateTime: "String", + DurationTypeEnum: "enum", + EditAccountInput:{ + userName:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + firstName:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + lastName:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + profilePicture:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + phoneNumber:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + gender:{ + type:"AccountGenderEnum", + array:false, + arrayRequired:false, + required:false + }, + birthDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + address:{ + type:"AddressInput", + array:false, + arrayRequired:false, + required:false + }, + language:{ + type:"AvailableTranslation", + array:false, + arrayRequired:false, + required:false + }, + settings:{ + type:"SettingsInput", + array:true, + arrayRequired:false, + required:true + } + }, + EditMessageInput:{ + message:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + EditWorkspaceInput:{ + title:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + teaser:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + cover:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + thumbnail:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + extraImages:{ + type:"ImageInput", + array:true, + arrayRequired:false, + required:true + }, + content:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + seo:{ + type:"SEOSimpleInput", + array:false, + arrayRequired:false, + required:false + }, + urls:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + workspaceType:{ + type:"WorkspaceTypeEnum", + array:false, + arrayRequired:false, + required:false + }, + pricingPerHour:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + currency:{ + type:"AvailableCurrency", + array:false, + arrayRequired:false, + required:true + }, + maxCapacity:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + minStay:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + maxStay:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + availability:{ + type:"AvailabilityInput", + array:false, + arrayRequired:false, + required:false + }, + mainImages:{ + type:"MainImage", + array:false, + arrayRequired:false, + required:true + }, + equipmentIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + featureIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + address:{ + type:"AddressStrictInput", + array:false, + arrayRequired:false, + required:false + } + }, + GeolocAddressSearchInput:{ + formattedAddress:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + radius:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + } + }, + GeolocSearchInput:{ + longitude:{ + type:"Float", + array:false, + arrayRequired:false, + required:true + }, + latitude:{ + type:"Float", + array:false, + arrayRequired:false, + required:true + }, + radius:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + } + }, + GetArgs:{ + limit:{ + type:"Int", + array:false, + arrayRequired:false, + required:true + }, + skip:{ + type:"Int", + array:false, + arrayRequired:false, + required:true + }, + sort:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + HoursInput:{ + from:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + to:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + ImageInput:{ + title:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + fileType:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + large:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + medium:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + small:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + InvoicingProvider: "enum", + JSONObject: "String", + LinkEmailInput:{ + email:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + password:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + idToken:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + ListEnum: "enum", + LoginInput:{ + email:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + password:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + MainImage:{ + main:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + one:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + two:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + three:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + four:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + } + }, + ModelLoadersEnum: "enum", + Mutation:{ + updateMe:{ + input:{ + type:"EditAccountInput", + array:false, + arrayRequired:false, + required:true + } + }, + updateMeEmail:{ + input:{ + type:"NewEmailInput", + array:false, + arrayRequired:false, + required:true + } + }, + updateMePassword:{ + input:{ + type:"NewPasswordInput", + array:false, + arrayRequired:false, + required:true + } + }, + resetPassword:{ + input:{ + type:"ResetPasswordInput", + array:false, + arrayRequired:false, + required:true + } + }, + registerGuest:{ + otherInfo:{ + type:"EditAccountInput", + array:false, + arrayRequired:false, + required:true + }, + input:{ + type:"LinkEmailInput", + array:false, + arrayRequired:false, + required:true + } + }, + register:{ + input:{ + type:"NewAccountInput", + array:false, + arrayRequired:false, + required:true + } + }, + accountsBillingInfosAddOne:{ + input:{ + type:"BillingAddressInput", + array:false, + arrayRequired:false, + required:true + } + }, + accountsBillingInfosEditOne:{ + input:{ + type:"BillingAddressInput", + array:false, + arrayRequired:false, + required:true + }, + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + accountsBillingInfosDeleteOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + accountsPaymentMethodsAddOne:{ + input:{ + type:"PaymentMethodInput", + array:false, + arrayRequired:false, + required:true + } + }, + accountsPaymentMethodsMarkasDefault:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + accountsPaymentMethodsDeleteOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + accountsFavoritesAddOne:{ + input:{ + type:"AddInput", + array:false, + arrayRequired:false, + required:true + } + }, + accountsFavoritesRemoveOne:{ + input:{ + type:"RemoveInput", + array:false, + arrayRequired:false, + required:true + } + }, + workspacesAddOne:{ + input:{ + type:"NewWorkspaceInput", + array:false, + arrayRequired:false, + required:true + } + }, + workspacesEditOne:{ + input:{ + type:"EditWorkspaceInput", + array:false, + arrayRequired:false, + required:true + }, + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + workspacesDeleteOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + finaliseCheckoutWithStripe:{ + input:{ + type:"CheckoutInput", + array:false, + arrayRequired:false, + required:true + } + }, + validateOrder:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + roomsCreateOne:{ + input:{ + type:"NewRoomInput", + array:false, + arrayRequired:false, + required:true + } + }, + roomsAddPeople:{ + input:{ + type:"AddPeopleToRoomInput", + array:false, + arrayRequired:false, + required:true + } + }, + roomsRemovePeople:{ + input:{ + type:"RemovePeopleFromRoomInput", + array:false, + arrayRequired:false, + required:true + } + }, + roomsMarkAllMessageAsRead:{ + roomId:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + reserveOneBooking:{ + input:{ + type:"BookingNewInputSchema", + array:false, + arrayRequired:false, + required:true + } + }, + roomsAddOneMessage:{ + input:{ + type:"NewMessageInput", + array:false, + arrayRequired:false, + required:true + } + }, + roomsEditOneMessage:{ + input:{ + type:"EditMessageInput", + array:false, + arrayRequired:false, + required:true + }, + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + roomsMarkMessageAsRead:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + roomsDeleteOneMessage:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + } + }, + NewAccountInput:{ + email:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + password:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + passwordConfirmation:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + userName:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + firstName:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + lastName:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + profilePicture:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + phoneNumber:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + gender:{ + type:"AccountGenderEnum", + array:false, + arrayRequired:false, + required:false + }, + birthDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + address:{ + type:"AddressInput", + array:false, + arrayRequired:false, + required:false + }, + language:{ + type:"AvailableTranslation", + array:false, + arrayRequired:false, + required:false + }, + settings:{ + type:"SettingsInput", + array:true, + arrayRequired:false, + required:true + }, + terms:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:true + } + }, + NewEmailInput:{ + newEmail:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + NewMessageInput:{ + roomId:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + message:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + file:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + } + }, + NewPasswordInput:{ + oldPassword:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + newPassword:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + newPasswordConfirmation:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + NewRoomInput:{ + accountIds:{ + type:"String", + array:true, + arrayRequired:true, + required:true + } + }, + NewWorkspaceInput:{ + title:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + teaser:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + cover:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + thumbnail:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + extraImages:{ + type:"ImageInput", + array:true, + arrayRequired:false, + required:true + }, + content:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + seo:{ + type:"SEOSimpleInput", + array:false, + arrayRequired:false, + required:false + }, + urls:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + workspaceType:{ + type:"WorkspaceTypeEnum", + array:false, + arrayRequired:false, + required:true + }, + pricingPerHour:{ + type:"Int", + array:false, + arrayRequired:false, + required:true + }, + currency:{ + type:"AvailableCurrency", + array:false, + arrayRequired:false, + required:true + }, + maxCapacity:{ + type:"Int", + array:false, + arrayRequired:false, + required:true + }, + minStay:{ + type:"Int", + array:false, + arrayRequired:false, + required:true + }, + maxStay:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + availability:{ + type:"AvailabilityInput", + array:false, + arrayRequired:false, + required:true + }, + mainImages:{ + type:"MainImage", + array:false, + arrayRequired:false, + required:true + }, + equipmentIds:{ + type:"String", + array:true, + arrayRequired:true, + required:true + }, + featureIds:{ + type:"String", + array:true, + arrayRequired:true, + required:true + }, + address:{ + type:"AddressStrictInput", + array:false, + arrayRequired:false, + required:true + } + }, + OrderBusinessStatusEnum: "enum", + OrderStatusEnum: "enum", + PaymentMethodInput:{ + sPayMethodId:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + nameOnCard:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + default:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + } + }, + PaymentProvider: "enum", + PaymentStatusEnum: "enum", + ProductRessourceEnum: "enum", + PromoType: "enum", + Query:{ + login:{ + refreshToken:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + creds:{ + type:"LoginInput", + array:false, + arrayRequired:false, + required:false + } + }, + magicLink:{ + input:{ + type:"ResetPasswordInput", + array:false, + arrayRequired:false, + required:true + } + }, + placesAutocomplete:{ + session:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + input:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + language:{ + type:"AvailableTranslation", + array:false, + arrayRequired:false, + required:false + }, + country:{ + type:"CountryCodesComponentEnum", + array:true, + arrayRequired:false, + required:true + } + }, + placesGeocode:{ + session:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + placeId:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + placesGeocodeFromAddress:{ + address:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + categoriesGetOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + categoriesGetMany:{ + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + ressourceType:{ + type:"RessourceEnum", + array:false, + arrayRequired:false, + required:true + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + }, + categoriesGetCount:{ + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + ressourceType:{ + type:"RessourceEnum", + array:false, + arrayRequired:false, + required:true + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + }, + listsGetOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + listsGetMany:{ + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + ressourceType:{ + type:"ListEnum", + array:false, + arrayRequired:false, + required:true + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + }, + listsGetCount:{ + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + ressourceType:{ + type:"ListEnum", + array:false, + arrayRequired:false, + required:true + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + }, + workspacesGetOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + workspacesGetMany:{ + geoSearch:{ + type:"GeolocSearchInput", + array:false, + arrayRequired:false, + required:false + }, + geoSearchWithAddress:{ + type:"GeolocAddressSearchInput", + array:false, + arrayRequired:false, + required:false + }, + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + dateFrom:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + dateTo:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + hoursFrom:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + hoursTo:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + workspaceType:{ + type:"WorkspaceTypeEnum", + array:false, + arrayRequired:false, + required:false + }, + workspaceTypes:{ + type:"WorkspaceTypeEnum", + array:true, + arrayRequired:false, + required:true + }, + priceMin:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + priceMax:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + maxCapacity:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + equipmentIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + featureIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + } + }, + workspacesGetCount:{ + geoSearch:{ + type:"GeolocSearchInput", + array:false, + arrayRequired:false, + required:false + }, + geoSearchWithAddress:{ + type:"GeolocAddressSearchInput", + array:false, + arrayRequired:false, + required:false + }, + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + dateFrom:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + dateTo:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + hoursFrom:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + hoursTo:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + workspaceType:{ + type:"WorkspaceTypeEnum", + array:false, + arrayRequired:false, + required:false + }, + workspaceTypes:{ + type:"WorkspaceTypeEnum", + array:true, + arrayRequired:false, + required:true + }, + priceMin:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + priceMax:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + maxCapacity:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + equipmentIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + featureIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + } + }, + workspacesSearchMany:{ + geoSearch:{ + type:"GeolocSearchInput", + array:false, + arrayRequired:false, + required:false + }, + geoSearchWithAddress:{ + type:"GeolocAddressSearchInput", + array:false, + arrayRequired:false, + required:false + }, + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + dateFrom:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + dateTo:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + hoursFrom:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + hoursTo:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + workspaceType:{ + type:"WorkspaceTypeEnum", + array:false, + arrayRequired:false, + required:false + }, + workspaceTypes:{ + type:"WorkspaceTypeEnum", + array:true, + arrayRequired:false, + required:true + }, + priceMin:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + priceMax:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + maxCapacity:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + equipmentIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + featureIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + } + }, + workspacesGetMine:{ + geoSearch:{ + type:"GeolocSearchInput", + array:false, + arrayRequired:false, + required:false + }, + geoSearchWithAddress:{ + type:"GeolocAddressSearchInput", + array:false, + arrayRequired:false, + required:false + }, + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + dateFrom:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + dateTo:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + hoursFrom:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + hoursTo:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + workspaceType:{ + type:"WorkspaceTypeEnum", + array:false, + arrayRequired:false, + required:false + }, + workspaceTypes:{ + type:"WorkspaceTypeEnum", + array:true, + arrayRequired:false, + required:true + }, + priceMin:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + priceMax:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + maxCapacity:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + equipmentIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + featureIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + } + }, + workspacesGetMineCount:{ + geoSearch:{ + type:"GeolocSearchInput", + array:false, + arrayRequired:false, + required:false + }, + geoSearchWithAddress:{ + type:"GeolocAddressSearchInput", + array:false, + arrayRequired:false, + required:false + }, + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + dateFrom:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + dateTo:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + hoursFrom:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + hoursTo:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + workspaceType:{ + type:"WorkspaceTypeEnum", + array:false, + arrayRequired:false, + required:false + }, + workspaceTypes:{ + type:"WorkspaceTypeEnum", + array:true, + arrayRequired:false, + required:true + }, + priceMin:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + priceMax:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + maxCapacity:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + equipmentIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + featureIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + } + }, + bookingsGetOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + bookingsGetMany:{ + startDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + endDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + rangeType:{ + type:"RangeType", + array:false, + arrayRequired:false, + required:false + }, + dateTabType:{ + type:"DateTabType", + array:false, + arrayRequired:false, + required:false + }, + dateRange:{ + type:"DateRangeComponentInput", + array:false, + arrayRequired:false, + required:false + }, + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + ressourceModel:{ + type:"BookingsRessourceEnum", + array:false, + arrayRequired:false, + required:false + }, + ressourceId:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + status:{ + type:"StatusEnum", + array:true, + arrayRequired:false, + required:true + } + }, + bookingsGetCount:{ + startDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + endDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + rangeType:{ + type:"RangeType", + array:false, + arrayRequired:false, + required:false + }, + dateTabType:{ + type:"DateTabType", + array:false, + arrayRequired:false, + required:false + }, + dateRange:{ + type:"DateRangeComponentInput", + array:false, + arrayRequired:false, + required:false + }, + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + ressourceModel:{ + type:"BookingsRessourceEnum", + array:false, + arrayRequired:false, + required:false + }, + ressourceId:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + status:{ + type:"StatusEnum", + array:true, + arrayRequired:false, + required:true + } + }, + myOrdersGetOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + myOrdersGetMany:{ + orderStatus:{ + type:"OrderStatusEnum", + array:false, + arrayRequired:false, + required:false + }, + paymentStatus:{ + type:"PaymentStatusEnum", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + myOrdersGetManyCount:{ + orderStatus:{ + type:"OrderStatusEnum", + array:false, + arrayRequired:false, + required:false + }, + paymentStatus:{ + type:"PaymentStatusEnum", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + roomsGetOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + roomsGetMany:{ + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + }, + roomsGetMessages:{ + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + } + }, + RangeType: "enum", + RemoveInput:{ + ressourceId:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + RemovePeopleFromRoomInput:{ + roomId:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + accountIds:{ + type:"String", + array:true, + arrayRequired:true, + required:true + } + }, + ResetPasswordInput:{ + email:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + RessourceEnum: "enum", + RoomModel:{ + getMessages:{ + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + } + }, + SEOSimpleInput:{ + title:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + description:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + keywords:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + thumbnail:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + } + }, + SettingsInput:{ + type:{ + type:"SettingsTypeEnum", + array:false, + arrayRequired:false, + required:true + }, + email:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + }, + pushNotifications:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + }, + sms:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + } + }, + SettingsTypeEnum: "enum", + SlotComponentInput:{ + startDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:true + }, + endDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:true + }, + startTime:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + endTime:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + StatusEnum: "enum", + WorkspaceModel:{ + getDistFromLocation:{ + longitude:{ + type:"Float", + array:false, + arrayRequired:false, + required:true + }, + latitude:{ + type:"Float", + array:false, + arrayRequired:false, + required:true + }, + radius:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + } + }, + getBookings:{ + ressourceModel:{ + type:"BookingsRessourceEnum", + array:false, + arrayRequired:false, + required:false + }, + ressourceId:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + status:{ + type:"StatusEnum", + array:true, + arrayRequired:false, + required:true + }, + startDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + endDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + rangeType:{ + type:"RangeType", + array:false, + arrayRequired:false, + required:false + }, + dateTabType:{ + type:"DateTabType", + array:false, + arrayRequired:false, + required:false + }, + dateRange:{ + type:"DateRangeComponentInput", + array:false, + arrayRequired:false, + required:false + }, + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + } + }, + WorkspaceTypeEnum: "enum" +} + +export const ReturnTypes: Record = { + AccountModel:{ + _id:"ID", + organisationId:"ID", + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID", + createdAt:"DateTime", + updatedAt:"DateTime", + r:"String", + w:"String", + d:"String", + types:"AccountTypeEnum", + organisationIds:"String", + email:"String", + userName:"String", + firstName:"String", + lastName:"String", + profilePicture:"ImageComponent", + phoneNumber:"String", + gender:"AccountGenderEnum", + birthDate:"DateTime", + getAge:"Int", + address:"AddressComponent", + language:"AvailableTranslation", + settings:"SettingsComponent", + paymentInfo:"AccountPaymentInfo", + payoutInfo:"AccountPayoutInfo", + terms:"Boolean", + favLikes:"FavoriteComponent", + getFavorites:"FavoriteComponent", + getLiked:"FavoriteComponent" + }, + AccountPaymentInfo:{ + stripeInfo:"StripeInfo", + paymentMethods:"PaymentMethodDetail", + billingInfos:"BillingAddressComponent" + }, + AccountPayoutInfo:{ + stripeInfo:"StripePayoutInfo" + }, + AddOnComponent:{ + addOnId:"String", + quantity:"Float" + }, + AddressComponent:{ + number:"String", + street:"String", + streetBis:"String", + floor:"String", + box:"String", + zip:"String", + state:"String", + city:"String", + country:"String" + }, + AddressStrictComponent:{ + number:"String", + street:"String", + zip:"String", + city:"String", + country:"String", + streetBis:"String", + floor:"String", + box:"String", + state:"String" + }, + AvailabilityComponent:{ + dates:"DateRangeComponent", + hours:"HourComponent", + exceptions:"DateExceptionComponent", + noWeekend:"Boolean" + }, + BillingAddressComponent:{ + firstName:"String", + lastName:"String", + address:"AddressStrictComponent", + email:"String", + default:"Boolean", + customTags:"JSONObject", + company:"String", + vatNumber:"String", + fiscalForm:"String", + _id:"String" + }, + BookingEngineSchema:{ + durationType:"DurationTypeEnum", + startToEnd:"DateRangeComponent", + slot:"SlotComponent", + capacity:"Int", + comments:"String", + status:"StatusEnum", + paths:"EnginePathComponent", + ownerId:"String", + dates:"DateRangeComponent", + owner:"AccountModel", + _id:"ID", + organisationId:"ID", + by:"MetaBy", + permissions:"MetaPermissions", + createdAt:"DateTime", + updatedAt:"DateTime", + workspace:"WorkspaceModel", + room:"RoomModel" + }, + BookingNewInput:{ + durationType:"DurationTypeEnum", + startToEnd:"DateRangeComponent", + slot:"SlotComponent", + capacity:"Int", + comments:"String", + ressourceModel:"BookingsRessourceEnum", + ressourceId:"String", + checkoutInfo:"CheckoutEngineInput" + }, + BookingSchema:{ + durationType:"DurationTypeEnum", + startToEnd:"DateRangeComponent", + slot:"SlotComponent", + capacity:"Int", + comments:"String", + status:"StatusEnum", + paths:"EnginePathComponent", + ownerId:"String", + dates:"DateRangeComponent", + _id:"ID", + organisationId:"ID", + by:"MetaBy", + permissions:"MetaPermissions", + createdAt:"DateTime", + updatedAt:"DateTime" + }, + BuyerInfoComponent:{ + firstName:"String", + lastName:"String", + email:"String", + company:"String", + vatNumber:"String", + address:"AddressComponent" + }, + CardInfo:{ + brand:"cardBrandEnum", + country:"String", + exp_month:"Int", + exp_year:"Int", + last4:"String" + }, + CategoryModel:{ + _id:"ID", + organisationId:"ID", + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID", + createdAt:"DateTime", + updatedAt:"DateTime", + r:"String", + w:"String", + d:"String", + title:"TranslatableComponent", + teaser:"TranslatableComponent", + cover:"ImageComponent", + thumbnail:"ImageComponent", + extraImages:"ImageComponent", + content:"TranslatableComponent", + seo:"SEOField", + urls:"TranslatableComponent", + ressourceType:"RessourceEnum", + colorCode:"String" + }, + CheckoutEngineInput:{ + contactInfo:"BuyerInfoComponent", + billingInfo:"BuyerInfoComponent", + token:"String" + }, + DateExceptionComponent:{ + dates:"DateRangeComponent", + hours:"HourComponent", + allDay:"Boolean" + }, + DateRangeComponent:{ + startDate:"DateTime", + endDate:"DateTime" + }, + EnginePathComponent:{ + ressourceModel:"ModelLoadersEnum", + ressourceId:"String" + }, + FavoriteComponent:{ + ressourceId:"String", + ressourceType:"RessourceEnum", + addedAt:"DateTime", + favorite:"Boolean", + liked:"Boolean" + }, + FirebaseTokenResult:{ + localId:"String", + email:"String", + displayName:"String", + idToken:"String", + registered:"Boolean", + refreshToken:"String", + expiresIn:"String" + }, + GeolocDistComponent:{ + calculated:"Float", + location:"LocComponent" + }, + HourComponent:{ + from:"String", + to:"String" + }, + IEngineSchema:{ + "...on BookingEngineSchema": "BookingEngineSchema", + "...on BookingSchema": "BookingSchema", + "...on OrderEngineSchema": "OrderEngineSchema", + _id:"ID", + organisationId:"ID", + paths:"EnginePathComponent", + by:"MetaBy", + permissions:"MetaPermissions", + createdAt:"DateTime", + updatedAt:"DateTime" + }, + ImageComponent:{ + title:"String", + fileType:"String", + large:"String", + medium:"String", + small:"String" + }, + InvoiceInfo:{ + provider:"InvoicingProvider", + invoiceId:"String", + invoiceNumber:"Float" + }, + LineItemSchema:{ + _id:"String", + productRessource:"ProductRessourceEnum", + productId:"String", + title:"String", + price:"Float", + salesPrice:"Float", + quantity:"Float", + vatClassId:"String", + parentId:"String", + addOns:"AddOnComponent", + localeInfo:"LocaleInfo", + promoId:"String", + finalPrice:"Float", + getProduct:"ProductUnion", + getLinePrice:"Float", + getLineVatPrice:"Float" + }, + ListModel:{ + _id:"ID", + organisationId:"ID", + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID", + createdAt:"DateTime", + updatedAt:"DateTime", + r:"String", + w:"String", + d:"String", + title:"TranslatableComponent", + teaser:"TranslatableComponent", + cover:"ImageComponent", + thumbnail:"ImageComponent", + extraImages:"ImageComponent", + content:"TranslatableComponent", + seo:"SEOField", + urls:"TranslatableComponent", + ressourceType:"ListEnum", + colorCode:"String" + }, + LocaleInfo:{ + countryCode:"String", + locale:"AvailableTranslation" + }, + LocComponent:{ + type:"String", + coordinates:"Float" + }, + MainImageComponent:{ + main:"ImageComponent", + one:"ImageComponent", + two:"ImageComponent", + three:"ImageComponent", + four:"ImageComponent" + }, + MessageModel:{ + _id:"ID", + organisationId:"ID", + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID", + createdAt:"DateTime", + updatedAt:"DateTime", + r:"String", + w:"String", + d:"String", + roomId:"String", + notified:"Boolean", + message:"String", + edited:"Boolean", + deleted:"Boolean", + file:"ImageComponent", + sentBy:"String", + readBy:"MessageReadBy" + }, + MessageReadBy:{ + accountId:"String", + readAt:"DateTime" + }, + MetaBy:{ + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID" + }, + MetaPermissions:{ + r:"String", + w:"String", + d:"String" + }, + Mutation:{ + updateMe:"AccountModel", + updateMeEmail:"AccountModel", + updateMePassword:"AccountModel", + resetPassword:"SimpleResult", + registerGuest:"AccountModel", + register:"AccountModel", + accountsBillingInfosAddOne:"BillingAddressComponent", + accountsBillingInfosEditOne:"BillingAddressComponent", + accountsBillingInfosDeleteOne:"String", + accountsPaymentMethodsAddOne:"PaymentMethodDetail", + accountsPaymentMethodsMarkasDefault:"PaymentMethodDetail", + accountsPaymentMethodsDeleteOne:"String", + accountsLinkWithStripe:"String", + accountsRefreshInfoStripe:"AccountModel", + accountsFavoritesAddOne:"FavoriteComponent", + accountsFavoritesRemoveOne:"SuccessResponse", + workspacesAddOne:"WorkspaceModel", + workspacesEditOne:"WorkspaceModel", + workspacesDeleteOne:"WorkspaceModel", + finaliseCheckoutWithStripe:"OrderEngineSchema", + removeCurrentOrder:"OrderEngineSchema", + validateOrder:"OrderEngineSchema", + roomsCreateOne:"RoomModel", + roomsAddPeople:"RoomModel", + roomsRemovePeople:"RoomModel", + roomsMarkAllMessageAsRead:"RoomModel", + reserveOneBooking:"OrderEngineSchema", + roomsAddOneMessage:"MessageModel", + roomsEditOneMessage:"MessageModel", + roomsMarkMessageAsRead:"MessageModel", + roomsDeleteOneMessage:"MessageModel" + }, + OrderEngineSchema:{ + lines:"LineItemSchema", + localeInfo:"LocaleInfo", + currency:"CurrencyEnum", + vatExempt:"Boolean", + accountId:"String", + contactInfo:"BuyerInfoComponent", + billingInfo:"BuyerInfoComponent", + promoId:"ID", + invoiceInfo:"InvoiceInfo", + providerOrderItems:"ProviderSchema", + orderName:"TranslatableComponent", + orderStatus:"OrderStatusEnum", + paymentIntent:"PaymentIntentInfo", + paymentStatus:"PaymentStatusEnum", + paymentInfo:"PaymentInfo", + subTotalPrice:"Float", + vatPrice:"Float", + finalPrice:"Float", + promo:"PromoModel", + _id:"ID", + organisationId:"ID", + paths:"EnginePathComponent", + by:"MetaBy", + permissions:"MetaPermissions", + createdAt:"DateTime", + updatedAt:"DateTime", + getBookingInput:"BookingNewInput" + }, + PaymentInfo:{ + provider:"PaymentProvider", + transactionId:"String" + }, + PaymentIntentInfo:{ + provider:"PaymentProvider", + stripePaymentIntentData:"StripePaymentIntentData" + }, + PaymentMethodDetail:{ + _id:"String", + sPayMethodId:"String", + default:"Boolean", + nameOnCard:"String", + type:"cardTypeEnum", + cardInfo:"CardInfo" + }, + PlaceComponent:{ + placeId:"String", + address:"AddressComponent", + loc:"LocComponent", + formattedAddress:"String" + }, + ProductUnion:{ + "...on WorkspaceModel":"WorkspaceModel" + }, + PromoModel:{ + _id:"ID", + organisationId:"ID", + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID", + createdAt:"DateTime", + updatedAt:"DateTime", + r:"String", + w:"String", + d:"String", + code:"String", + description:"TranslatableComponent", + type:"PromoType", + value:"Float", + validity:"DateTime", + cummulable:"Boolean", + usageLimit:"Float", + usage:"Float" + }, + ProviderSchema:{ + _id:"String", + organisationId:"String", + orderStatus:"OrderBusinessStatusEnum", + lines:"LineItemSchema" + }, + Query:{ + me:"AccountModel", + login:"FirebaseTokenResult", + magicLink:"SimpleResult", + placesAutocomplete:"JSONObject", + placesGeocode:"JSONObject", + placesGeocodeFromAddress:"JSONObject", + categoriesGetOne:"CategoryModel", + categoriesGetMany:"CategoryModel", + categoriesGetCount:"Float", + listsGetOne:"ListModel", + listsGetMany:"ListModel", + listsGetCount:"Float", + accountsDashboardLinkStripe:"String", + workspacesGetOne:"WorkspaceModel", + workspacesGetMany:"WorkspaceModel", + workspacesGetCount:"Float", + workspacesSearchMany:"WorkspaceModel", + workspacesGetMine:"WorkspaceModel", + workspacesGetMineCount:"Float", + bookingsGetOne:"BookingEngineSchema", + bookingsGetMany:"BookingEngineSchema", + bookingsGetCount:"Float", + myCurrentOrder:"OrderEngineSchema", + myOrdersGetOne:"OrderEngineSchema", + myOrdersGetMany:"OrderEngineSchema", + myOrdersGetManyCount:"Float", + roomsGetOne:"RoomModel", + roomsGetMany:"RoomModel", + roomsGetMessages:"MessageModel" + }, + RoomModel:{ + _id:"ID", + organisationId:"ID", + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID", + createdAt:"DateTime", + updatedAt:"DateTime", + r:"String", + w:"String", + d:"String", + title:"String", + stats:"RoomsStats", + getAccounts:"AccountModel", + getMessages:"MessageModel", + lastMessage:"DateTime" + }, + RoomsStats:{ + unreadCounts:"UnreadCount" + }, + SEOField:{ + title:"TranslatableComponent", + description:"TranslatableComponent", + keywords:"TranslatableComponent", + thumbnail:"ImageComponent" + }, + SEOSimpleField:{ + title:"String", + description:"String", + keywords:"String", + thumbnail:"ImageComponent" + }, + SettingsComponent:{ + type:"SettingsTypeEnum", + email:"Boolean", + pushNotifications:"Boolean", + sms:"Boolean" + }, + SimpleResult:{ + message:"String" + }, + SlotComponent:{ + startDate:"DateTime", + endDate:"DateTime", + startTime:"String", + endTime:"String" + }, + StripeInfo:{ + customerId:"String" + }, + StripePaymentIntentData:{ + id:"String", + client_secret:"String", + currency:"String", + customer:"String", + status:"String" + }, + StripePayoutInfo:{ + accountId:"String", + chargesEnabled:"Boolean", + payoutsEnabled:"Boolean", + detailsSubmitted:"Boolean" + }, + SuccessResponse:{ + success:"Boolean" + }, + TranslatableComponent:{ + en:"String", + fr:"String", + nl:"String", + de:"String" + }, + UnreadCount:{ + accountId:"String", + count:"Float" + }, + WorkspaceModel:{ + _id:"ID", + organisationId:"ID", + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID", + createdAt:"DateTime", + updatedAt:"DateTime", + r:"String", + w:"String", + d:"String", + title:"String", + teaser:"String", + cover:"ImageComponent", + thumbnail:"ImageComponent", + extraImages:"ImageComponent", + content:"String", + seo:"SEOSimpleField", + urls:"String", + place:"PlaceComponent", + dist:"GeolocDistComponent", + getDistFromLocation:"Float", + workspaceType:"WorkspaceTypeEnum", + pricingPerHour:"Int", + currency:"AvailableCurrency", + maxCapacity:"Int", + minStay:"Int", + maxStay:"Int", + availability:"AvailabilityComponent", + mainImages:"MainImageComponent", + equipmentIds:"String", + featureIds:"String", + address:"AddressComponent", + finalPrice:"Float", + getEquipments:"ListModel", + getFeatures:"ListModel", + getBookings:"BookingSchema", + getOwner:"WorkspaceOwnerComponent" + }, + WorkspaceOwnerComponent:{ + firstName:"String", + lastName:"String" + } +} + +export class GraphQLError extends Error { + constructor(public response: GraphQLResponse) { + super(""); + console.error(response); + } + toString() { + return "GraphQL Response Error"; + } + } + + + +export type UnwrapPromise = T extends Promise ? R : T; +export type ZeusState Promise> = NonNullable< + UnwrapPromise> +>; +export type ZeusHook< + T extends ( + ...args: any[] + ) => Record Promise>, + N extends keyof ReturnType +> = ZeusState[N]>; + +type Func

= (...args: P) => R; +type AnyFunc = Func; + +type WithTypeNameValue = T & { + __typename?: true; +}; + +type AliasType = WithTypeNameValue & { + __alias?: Record>; +}; + +type NotUndefined = T extends undefined ? never : T; + +export type ResolverType = NotUndefined; + +export type ArgsType = F extends Func ? P : never; + +interface GraphQLResponse { + data?: Record; + errors?: Array<{ + message: string; + }>; +} + +export type ValuesOf = T[keyof T]; + +export type MapResolve = SRC extends { + __interface: infer INTERFACE; + __resolve: Record & infer IMPLEMENTORS; + } + ? + ValuesOf<{ + [k in (keyof SRC['__resolve'] & keyof DST)]: ({ + [rk in (keyof SRC['__resolve'][k] & keyof DST[k])]: LastMapTypeSRCResolver + } & { + __typename: SRC['__resolve'][k]['__typename'] + }) + }> + : + never; + +export type MapInterface = SRC extends { + __interface: infer INTERFACE; + __resolve: Record & infer IMPLEMENTORS; + } + ? + (MapResolve extends never ? {} : MapResolve) & { + [k in (keyof SRC['__interface'] & keyof DST)]: LastMapTypeSRCResolver +} : never; + +export type ValueToUnion = T extends { + __typename: infer R; +} + ? { + [P in keyof Omit]: T[P] & { + __typename: R; + }; + } + : T; + +export type ObjectToUnion = { + [P in keyof T]: T[P]; +}[keyof T]; + +type Anify = { [P in keyof T]?: any }; + + +type LastMapTypeSRCResolver = SRC extends undefined + ? undefined + : SRC extends Array + ? LastMapTypeSRCResolver[] + : SRC extends { __interface: any; __resolve: any } + ? MapInterface + : SRC extends { __union: any; __resolve: infer RESOLVE } + ? ObjectToUnion>> + : DST extends boolean + ? SRC + : MapType; + +export type MapType, DST> = DST extends boolean + ? SRC + : DST extends { + __alias: any; + } + ? { + [A in keyof DST["__alias"]]: Required extends Anify< + DST["__alias"][A] + > + ? MapType, DST["__alias"][A]> + : never; + } & + { + [Key in keyof Omit]: DST[Key] extends [ + any, + infer PAYLOAD + ] + ? LastMapTypeSRCResolver + : LastMapTypeSRCResolver; + } + : { + [Key in keyof DST]: DST[Key] extends [any, infer PAYLOAD] + ? LastMapTypeSRCResolver + : LastMapTypeSRCResolver; + }; + +type OperationToGraphQL = (o: Z | V, variables?: Record) => Promise>; + +type CastToGraphQL = ( + resultOfYourQuery: any +) => (o: Z | V) => MapType; + +type fetchOptions = ArgsType; + +export type SelectionFunction = (t: T | V) => T; +type FetchFunction = (query: string, variables?: Record) => Promise; + + + +export const ZeusSelect = () => ((t: any) => t) as SelectionFunction; + +export const ScalarResolver = (scalar: string, value: any) => { + switch (scalar) { + case 'String': + return `${JSON.stringify(value)}`; + case 'Int': + return `${value}`; + case 'Float': + return `${value}`; + case 'Boolean': + return `${value}`; + case 'ID': + return `"${value}"`; + case 'enum': + return `${value}`; + case 'scalar': + return `${value}`; + default: + return false; + } +}; + + +export const TypesPropsResolver = ({ + value, + type, + name, + key, + blockArrays +}: { + value: any; + type: string; + name: string; + key?: string; + blockArrays?: boolean; +}): string => { + if (value === null) { + return `null`; + } + let resolvedValue = AllTypesProps[type][name]; + if (key) { + resolvedValue = resolvedValue[key]; + } + if (!resolvedValue) { + throw new Error(`Cannot resolve ${type} ${name}${key ? ` ${key}` : ''}`) + } + const typeResolved = resolvedValue.type; + const isArray = resolvedValue.array; + const isArrayRequired = resolvedValue.arrayRequired; + if (typeof value === 'string' && value.startsWith(`ZEUS_VAR$`)) { + const isRequired = resolvedValue.required ? '!' : ''; + let t = `${typeResolved}`; + if (isArray) { + if (isRequired) { + t = `${t}!`; + } + t = `[${t}]`; + if(isArrayRequired){ + t = `${t}!`; + } + }else{ + if (isRequired) { + t = `${t}!`; + } + } + return `\$${value.split(`ZEUS_VAR$`)[1]}__ZEUS_VAR__${t}`; + } + if (isArray && !blockArrays) { + return `[${value + .map((v: any) => TypesPropsResolver({ value: v, type, name, key, blockArrays: true })) + .join(',')}]`; + } + const reslovedScalar = ScalarResolver(typeResolved, value); + if (!reslovedScalar) { + const resolvedType = AllTypesProps[typeResolved]; + if (typeof resolvedType === 'object') { + const argsKeys = Object.keys(resolvedType); + return `{${argsKeys + .filter((ak) => value[ak] !== undefined) + .map( + (ak) => `${ak}:${TypesPropsResolver({ value: value[ak], type: typeResolved, name: ak })}` + )}}`; + } + return ScalarResolver(AllTypesProps[typeResolved], value) as string; + } + return reslovedScalar; +}; + + +const isArrayFunction = ( + parent: string[], + a: any[] +) => { + const [values, r] = a; + const [mainKey, key, ...keys] = parent; + const keyValues = Object.keys(values).filter((k) => typeof values[k] !== 'undefined'); + + if (!keys.length) { + return keyValues.length > 0 + ? `(${keyValues + .map( + (v) => + `${v}:${TypesPropsResolver({ + value: values[v], + type: mainKey, + name: key, + key: v + })}` + ) + .join(',')})${r ? traverseToSeekArrays(parent, r) : ''}` + : traverseToSeekArrays(parent, r); + } + + const [typeResolverKey] = keys.splice(keys.length - 1, 1); + let valueToResolve = ReturnTypes[mainKey][key]; + for (const k of keys) { + valueToResolve = ReturnTypes[valueToResolve][k]; + } + + const argumentString = + keyValues.length > 0 + ? `(${keyValues + .map( + (v) => + `${v}:${TypesPropsResolver({ + value: values[v], + type: valueToResolve, + name: typeResolverKey, + key: v + })}` + ) + .join(',')})${r ? traverseToSeekArrays(parent, r) : ''}` + : traverseToSeekArrays(parent, r); + return argumentString; +}; + + +const resolveKV = (k: string, v: boolean | string | { [x: string]: boolean | string }) => + typeof v === 'boolean' ? k : typeof v === 'object' ? `${k}{${objectToTree(v)}}` : `${k}${v}`; + + +const objectToTree = (o: { [x: string]: boolean | string }): string => + `{${Object.keys(o).map((k) => `${resolveKV(k, o[k])}`).join(' ')}}`; + + +const traverseToSeekArrays = (parent: string[], a?: any): string => { + if (!a) return ''; + if (Object.keys(a).length === 0) { + return ''; + } + let b: Record = {}; + if (Array.isArray(a)) { + return isArrayFunction([...parent], a); + } else { + if (typeof a === 'object') { + Object.keys(a) + .filter((k) => typeof a[k] !== 'undefined') + .map((k) => { + if (k === '__alias') { + Object.keys(a[k]).map((aliasKey) => { + const aliasOperations = a[k][aliasKey]; + const aliasOperationName = Object.keys(aliasOperations)[0]; + const aliasOperation = aliasOperations[aliasOperationName]; + b[ + `${aliasOperationName}__alias__${aliasKey}: ${aliasOperationName}` + ] = traverseToSeekArrays([...parent, aliasOperationName], aliasOperation); + }); + } else { + b[k] = traverseToSeekArrays([...parent, k], a[k]); + } + }); + } else { + return ''; + } + } + return objectToTree(b); +}; + + +const buildQuery = (type: string, a?: Record) => + traverseToSeekArrays([type], a); + + +const inspectVariables = (query: string) => { + const regex = /\$\b\w*__ZEUS_VAR__\[?[^!^\]^\s^,^\)^\}]*[!]?[\]]?[!]?/g; + let result; + const AllVariables: string[] = []; + while ((result = regex.exec(query))) { + if (AllVariables.includes(result[0])) { + continue; + } + AllVariables.push(result[0]); + } + if (!AllVariables.length) { + return query; + } + let filteredQuery = query; + AllVariables.forEach((variable) => { + while (filteredQuery.includes(variable)) { + filteredQuery = filteredQuery.replace(variable, variable.split('__ZEUS_VAR__')[0]); + } + }); + return `(${AllVariables.map((a) => a.split('__ZEUS_VAR__')) + .map(([variableName, variableType]) => `${variableName}:${variableType}`) + .join(', ')})${filteredQuery}`; +}; + + +const queryConstruct = (t: 'query' | 'mutation' | 'subscription', tName: string) => (o: Record) => + `${t.toLowerCase()}${inspectVariables(buildQuery(tName, o))}`; + + +const fullChainConstruct = (fn: FetchFunction) => (t: 'query' | 'mutation' | 'subscription', tName: string) => ( + o: Record, + variables?: Record, +) => fn(queryConstruct(t, tName)(o), variables).then((r:any) => { + seekForAliases(r) + return r +}); + + +const seekForAliases = (response: any) => { + const traverseAlias = (value: any) => { + if (Array.isArray(value)) { + value.forEach(seekForAliases); + } else { + if (typeof value === 'object') { + seekForAliases(value); + } + } + }; + if (typeof response === 'object' && response) { + const keys = Object.keys(response); + if (keys.length < 1) { + return; + } + keys.forEach((k) => { + const value = response[k]; + if (k.indexOf('__alias__') !== -1) { + const [operation, alias] = k.split('__alias__'); + response[alias] = { + [operation]: value, + }; + delete response[k]; + } + traverseAlias(value); + }); + } +}; + + +export const $ = (t: TemplateStringsArray): any => `ZEUS_VAR$${t.join('')}`; + + +const handleFetchResponse = ( + response: Parameters['then']>[0], Function>>[0] +): Promise => { + if (!response.ok) { + return new Promise((_, reject) => { + response.text().then(text => { + try { reject(JSON.parse(text)); } + catch (err) { reject(text); } + }).catch(reject); + }); + } + return response.json(); +}; + +const apiFetch = (options: fetchOptions) => (query: string, variables: Record = {}) => { + let fetchFunction = fetch; + let queryString = query; + let fetchOptions = options[1] || {}; + if (fetchOptions.method && fetchOptions.method === 'GET') { + queryString = encodeURIComponent(query); + return fetchFunction(`${options[0]}?query=${queryString}`, fetchOptions) + .then(handleFetchResponse) + .then((response: GraphQLResponse) => { + if (response.errors) { + throw new GraphQLError(response); + } + return response.data; + }); + } + return fetchFunction(`${options[0]}`, { + body: JSON.stringify({ query: queryString, variables }), + method: 'POST', + headers: { + 'Content-Type': 'application/json' + }, + ...fetchOptions + }) + .then(handleFetchResponse) + .then((response: GraphQLResponse) => { + if (response.errors) { + throw new GraphQLError(response); + } + return response.data; + }); + }; + + + +export const Thunder = (fn: FetchFunction) => ({ + query: ((o: any, variables) => + fullChainConstruct(fn)('query', 'Query')(o, variables).then( + (response: any) => response + )) as OperationToGraphQL, +mutation: ((o: any, variables) => + fullChainConstruct(fn)('mutation', 'Mutation')(o, variables).then( + (response: any) => response + )) as OperationToGraphQL +}); + +export const Chain = (...options: fetchOptions) => ({ + query: ((o: any, variables) => + fullChainConstruct(apiFetch(options))('query', 'Query')(o, variables).then( + (response: any) => response + )) as OperationToGraphQL, +mutation: ((o: any, variables) => + fullChainConstruct(apiFetch(options))('mutation', 'Mutation')(o, variables).then( + (response: any) => response + )) as OperationToGraphQL +}); +export const Zeus = { + query: (o:ValueTypes["Query"]) => queryConstruct('query', 'Query')(o), +mutation: (o:ValueTypes["Mutation"]) => queryConstruct('mutation', 'Mutation')(o) +}; +export const Cast = { + query: ((o: any) => (_: any) => o) as CastToGraphQL< + ValueTypes["Query"], + Query +>, +mutation: ((o: any) => (_: any) => o) as CastToGraphQL< + ValueTypes["Mutation"], + Mutation +> +}; +export const Selectors = { + query: ZeusSelect(), +mutation: ZeusSelect() +}; + + +export const Gql = Chain('http://localhost:4000') \ No newline at end of file diff --git a/__tests/app/helpers/zeus/const.ts b/__tests/app/helpers/zeus/const.ts new file mode 100644 index 0000000..b2ec4f7 --- /dev/null +++ b/__tests/app/helpers/zeus/const.ts @@ -0,0 +1,3093 @@ +/* eslint-disable */ + +export const AllTypesProps: Record = { + Query:{ + login:{ + refreshToken:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + creds:{ + type:"LoginInput", + array:false, + arrayRequired:false, + required:false + } + }, + magicLink:{ + input:{ + type:"ResetPasswordInput", + array:false, + arrayRequired:false, + required:true + } + }, + placesAutocomplete:{ + session:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + input:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + language:{ + type:"AvailableTranslation", + array:false, + arrayRequired:false, + required:false + }, + country:{ + type:"CountryCodesComponentEnum", + array:true, + arrayRequired:false, + required:true + } + }, + placesGeocode:{ + session:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + placeId:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + placesGeocodeFromAddress:{ + address:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + categoriesGetOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + categoriesGetMany:{ + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + ressourceType:{ + type:"RessourceEnum", + array:false, + arrayRequired:false, + required:true + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + }, + categoriesGetCount:{ + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + ressourceType:{ + type:"RessourceEnum", + array:false, + arrayRequired:false, + required:true + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + }, + listsGetOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + listsGetMany:{ + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + ressourceType:{ + type:"ListEnum", + array:false, + arrayRequired:false, + required:true + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + }, + listsGetCount:{ + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + ressourceType:{ + type:"ListEnum", + array:false, + arrayRequired:false, + required:true + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + }, + workspacesGetOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + workspacesGetMany:{ + geoSearch:{ + type:"GeolocSearchInput", + array:false, + arrayRequired:false, + required:false + }, + geoSearchWithAddress:{ + type:"GeolocAddressSearchInput", + array:false, + arrayRequired:false, + required:false + }, + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + dateFrom:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + dateTo:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + hoursFrom:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + hoursTo:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + workspaceType:{ + type:"WorkspaceTypeEnum", + array:false, + arrayRequired:false, + required:false + }, + workspaceTypes:{ + type:"WorkspaceTypeEnum", + array:true, + arrayRequired:false, + required:true + }, + priceMin:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + priceMax:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + maxCapacity:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + equipmentIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + featureIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + } + }, + workspacesGetCount:{ + geoSearch:{ + type:"GeolocSearchInput", + array:false, + arrayRequired:false, + required:false + }, + geoSearchWithAddress:{ + type:"GeolocAddressSearchInput", + array:false, + arrayRequired:false, + required:false + }, + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + dateFrom:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + dateTo:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + hoursFrom:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + hoursTo:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + workspaceType:{ + type:"WorkspaceTypeEnum", + array:false, + arrayRequired:false, + required:false + }, + workspaceTypes:{ + type:"WorkspaceTypeEnum", + array:true, + arrayRequired:false, + required:true + }, + priceMin:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + priceMax:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + maxCapacity:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + equipmentIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + featureIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + } + }, + workspacesSearchMany:{ + geoSearch:{ + type:"GeolocSearchInput", + array:false, + arrayRequired:false, + required:false + }, + geoSearchWithAddress:{ + type:"GeolocAddressSearchInput", + array:false, + arrayRequired:false, + required:false + }, + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + dateFrom:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + dateTo:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + hoursFrom:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + hoursTo:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + workspaceType:{ + type:"WorkspaceTypeEnum", + array:false, + arrayRequired:false, + required:false + }, + workspaceTypes:{ + type:"WorkspaceTypeEnum", + array:true, + arrayRequired:false, + required:true + }, + priceMin:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + priceMax:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + maxCapacity:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + equipmentIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + featureIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + } + }, + workspacesGetMine:{ + geoSearch:{ + type:"GeolocSearchInput", + array:false, + arrayRequired:false, + required:false + }, + geoSearchWithAddress:{ + type:"GeolocAddressSearchInput", + array:false, + arrayRequired:false, + required:false + }, + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + dateFrom:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + dateTo:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + hoursFrom:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + hoursTo:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + workspaceType:{ + type:"WorkspaceTypeEnum", + array:false, + arrayRequired:false, + required:false + }, + workspaceTypes:{ + type:"WorkspaceTypeEnum", + array:true, + arrayRequired:false, + required:true + }, + priceMin:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + priceMax:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + maxCapacity:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + equipmentIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + featureIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + } + }, + workspacesGetMineCount:{ + geoSearch:{ + type:"GeolocSearchInput", + array:false, + arrayRequired:false, + required:false + }, + geoSearchWithAddress:{ + type:"GeolocAddressSearchInput", + array:false, + arrayRequired:false, + required:false + }, + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + dateFrom:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + dateTo:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + hoursFrom:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + hoursTo:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + workspaceType:{ + type:"WorkspaceTypeEnum", + array:false, + arrayRequired:false, + required:false + }, + workspaceTypes:{ + type:"WorkspaceTypeEnum", + array:true, + arrayRequired:false, + required:true + }, + priceMin:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + priceMax:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + maxCapacity:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + equipmentIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + featureIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + } + }, + bookingsGetOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + bookingsGetMany:{ + startDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + endDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + rangeType:{ + type:"RangeType", + array:false, + arrayRequired:false, + required:false + }, + dateTabType:{ + type:"DateTabType", + array:false, + arrayRequired:false, + required:false + }, + dateRange:{ + type:"DateRangeComponentInput", + array:false, + arrayRequired:false, + required:false + }, + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + ressourceModel:{ + type:"BookingsRessourceEnum", + array:false, + arrayRequired:false, + required:false + }, + ressourceId:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + status:{ + type:"StatusEnum", + array:true, + arrayRequired:false, + required:true + } + }, + bookingsGetCount:{ + startDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + endDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + rangeType:{ + type:"RangeType", + array:false, + arrayRequired:false, + required:false + }, + dateTabType:{ + type:"DateTabType", + array:false, + arrayRequired:false, + required:false + }, + dateRange:{ + type:"DateRangeComponentInput", + array:false, + arrayRequired:false, + required:false + }, + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + ressourceModel:{ + type:"BookingsRessourceEnum", + array:false, + arrayRequired:false, + required:false + }, + ressourceId:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + status:{ + type:"StatusEnum", + array:true, + arrayRequired:false, + required:true + } + }, + myOrdersGetOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + myOrdersGetMany:{ + orderStatus:{ + type:"OrderStatusEnum", + array:false, + arrayRequired:false, + required:false + }, + paymentStatus:{ + type:"PaymentStatusEnum", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + myOrdersGetManyCount:{ + orderStatus:{ + type:"OrderStatusEnum", + array:false, + arrayRequired:false, + required:false + }, + paymentStatus:{ + type:"PaymentStatusEnum", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + roomsGetOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + roomsGetMany:{ + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + }, + roomsGetMessages:{ + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + }, + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + } + }, + DateTime: "String", + AccountTypeEnum: "enum", + AccountGenderEnum: "enum", + AvailableTranslation: "enum", + SettingsTypeEnum: "enum", + cardTypeEnum: "enum", + cardBrandEnum: "enum", + JSONObject: "String", + RessourceEnum: "enum", + LoginInput:{ + email:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + password:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + ResetPasswordInput:{ + email:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + CountryCodesComponentEnum: "enum", + GetArgs:{ + limit:{ + type:"Int", + array:false, + arrayRequired:false, + required:true + }, + skip:{ + type:"Int", + array:false, + arrayRequired:false, + required:true + }, + sort:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + ListEnum: "enum", + WorkspaceModel:{ + getDistFromLocation:{ + longitude:{ + type:"Float", + array:false, + arrayRequired:false, + required:true + }, + latitude:{ + type:"Float", + array:false, + arrayRequired:false, + required:true + }, + radius:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + } + }, + getBookings:{ + ressourceModel:{ + type:"BookingsRessourceEnum", + array:false, + arrayRequired:false, + required:false + }, + ressourceId:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + status:{ + type:"StatusEnum", + array:true, + arrayRequired:false, + required:true + }, + startDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + endDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + rangeType:{ + type:"RangeType", + array:false, + arrayRequired:false, + required:false + }, + dateTabType:{ + type:"DateTabType", + array:false, + arrayRequired:false, + required:false + }, + dateRange:{ + type:"DateRangeComponentInput", + array:false, + arrayRequired:false, + required:false + }, + _ids:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + search:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + afterCreatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + afterUpdatedAt:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + } + }, + WorkspaceTypeEnum: "enum", + AvailableCurrency: "enum", + ModelLoadersEnum: "enum", + DurationTypeEnum: "enum", + StatusEnum: "enum", + BookingsRessourceEnum: "enum", + RangeType: "enum", + DateTabType: "enum", + DateRangeComponentInput:{ + startDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:true + }, + endDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:true + } + }, + GeolocSearchInput:{ + longitude:{ + type:"Float", + array:false, + arrayRequired:false, + required:true + }, + latitude:{ + type:"Float", + array:false, + arrayRequired:false, + required:true + }, + radius:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + } + }, + GeolocAddressSearchInput:{ + formattedAddress:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + radius:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + } + }, + RoomModel:{ + getMessages:{ + pagination:{ + type:"GetArgs", + array:false, + arrayRequired:false, + required:false + } + } + }, + ProductRessourceEnum: "enum", + CurrencyEnum: "enum", + InvoicingProvider: "enum", + OrderBusinessStatusEnum: "enum", + OrderStatusEnum: "enum", + PaymentProvider: "enum", + PaymentStatusEnum: "enum", + PromoType: "enum", + Mutation:{ + updateMe:{ + input:{ + type:"EditAccountInput", + array:false, + arrayRequired:false, + required:true + } + }, + updateMeEmail:{ + input:{ + type:"NewEmailInput", + array:false, + arrayRequired:false, + required:true + } + }, + updateMePassword:{ + input:{ + type:"NewPasswordInput", + array:false, + arrayRequired:false, + required:true + } + }, + resetPassword:{ + input:{ + type:"ResetPasswordInput", + array:false, + arrayRequired:false, + required:true + } + }, + registerGuest:{ + otherInfo:{ + type:"EditAccountInput", + array:false, + arrayRequired:false, + required:true + }, + input:{ + type:"LinkEmailInput", + array:false, + arrayRequired:false, + required:true + } + }, + register:{ + input:{ + type:"NewAccountInput", + array:false, + arrayRequired:false, + required:true + } + }, + accountsBillingInfosAddOne:{ + input:{ + type:"BillingAddressInput", + array:false, + arrayRequired:false, + required:true + } + }, + accountsBillingInfosEditOne:{ + input:{ + type:"BillingAddressInput", + array:false, + arrayRequired:false, + required:true + }, + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + accountsBillingInfosDeleteOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + accountsPaymentMethodsAddOne:{ + input:{ + type:"PaymentMethodInput", + array:false, + arrayRequired:false, + required:true + } + }, + accountsPaymentMethodsMarkasDefault:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + accountsPaymentMethodsDeleteOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + accountsFavoritesAddOne:{ + input:{ + type:"AddInput", + array:false, + arrayRequired:false, + required:true + } + }, + accountsFavoritesRemoveOne:{ + input:{ + type:"RemoveInput", + array:false, + arrayRequired:false, + required:true + } + }, + workspacesAddOne:{ + input:{ + type:"NewWorkspaceInput", + array:false, + arrayRequired:false, + required:true + } + }, + workspacesEditOne:{ + input:{ + type:"EditWorkspaceInput", + array:false, + arrayRequired:false, + required:true + }, + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + workspacesDeleteOne:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + finaliseCheckoutWithStripe:{ + input:{ + type:"CheckoutInput", + array:false, + arrayRequired:false, + required:true + } + }, + validateOrder:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + roomsCreateOne:{ + input:{ + type:"NewRoomInput", + array:false, + arrayRequired:false, + required:true + } + }, + roomsAddPeople:{ + input:{ + type:"AddPeopleToRoomInput", + array:false, + arrayRequired:false, + required:true + } + }, + roomsRemovePeople:{ + input:{ + type:"RemovePeopleFromRoomInput", + array:false, + arrayRequired:false, + required:true + } + }, + roomsMarkAllMessageAsRead:{ + roomId:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + reserveOneBooking:{ + input:{ + type:"BookingNewInputSchema", + array:false, + arrayRequired:false, + required:true + } + }, + roomsAddOneMessage:{ + input:{ + type:"NewMessageInput", + array:false, + arrayRequired:false, + required:true + } + }, + roomsEditOneMessage:{ + input:{ + type:"EditMessageInput", + array:false, + arrayRequired:false, + required:true + }, + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + roomsMarkMessageAsRead:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + roomsDeleteOneMessage:{ + id:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + } + }, + EditAccountInput:{ + userName:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + firstName:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + lastName:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + profilePicture:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + phoneNumber:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + gender:{ + type:"AccountGenderEnum", + array:false, + arrayRequired:false, + required:false + }, + birthDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + address:{ + type:"AddressInput", + array:false, + arrayRequired:false, + required:false + }, + language:{ + type:"AvailableTranslation", + array:false, + arrayRequired:false, + required:false + }, + settings:{ + type:"SettingsInput", + array:true, + arrayRequired:false, + required:true + } + }, + ImageInput:{ + title:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + fileType:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + large:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + medium:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + small:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + AddressInput:{ + number:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + street:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + streetBis:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + floor:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + box:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + zip:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + state:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + city:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + country:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + SettingsInput:{ + type:{ + type:"SettingsTypeEnum", + array:false, + arrayRequired:false, + required:true + }, + email:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + }, + pushNotifications:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + }, + sms:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + } + }, + NewEmailInput:{ + newEmail:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + NewPasswordInput:{ + oldPassword:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + newPassword:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + newPasswordConfirmation:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + LinkEmailInput:{ + email:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + password:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + idToken:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + NewAccountInput:{ + email:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + password:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + passwordConfirmation:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + userName:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + firstName:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + lastName:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + profilePicture:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + phoneNumber:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + gender:{ + type:"AccountGenderEnum", + array:false, + arrayRequired:false, + required:false + }, + birthDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:false + }, + address:{ + type:"AddressInput", + array:false, + arrayRequired:false, + required:false + }, + language:{ + type:"AvailableTranslation", + array:false, + arrayRequired:false, + required:false + }, + settings:{ + type:"SettingsInput", + array:true, + arrayRequired:false, + required:true + }, + terms:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:true + } + }, + BillingAddressInput:{ + firstName:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + lastName:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + address:{ + type:"AddressStrictInput", + array:false, + arrayRequired:false, + required:true + }, + email:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + default:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + }, + customTags:{ + type:"JSONObject", + array:false, + arrayRequired:false, + required:false + }, + company:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + vatNumber:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + fiscalForm:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + AddressStrictInput:{ + number:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + street:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + zip:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + city:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + country:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + streetBis:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + floor:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + box:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + state:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + PaymentMethodInput:{ + sPayMethodId:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + nameOnCard:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + default:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + } + }, + AddInput:{ + ressourceId:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + ressourceType:{ + type:"RessourceEnum", + array:false, + arrayRequired:false, + required:true + } + }, + RemoveInput:{ + ressourceId:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + NewWorkspaceInput:{ + title:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + teaser:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + cover:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + thumbnail:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + extraImages:{ + type:"ImageInput", + array:true, + arrayRequired:false, + required:true + }, + content:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + seo:{ + type:"SEOSimpleInput", + array:false, + arrayRequired:false, + required:false + }, + urls:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + workspaceType:{ + type:"WorkspaceTypeEnum", + array:false, + arrayRequired:false, + required:true + }, + pricingPerHour:{ + type:"Int", + array:false, + arrayRequired:false, + required:true + }, + currency:{ + type:"AvailableCurrency", + array:false, + arrayRequired:false, + required:true + }, + maxCapacity:{ + type:"Int", + array:false, + arrayRequired:false, + required:true + }, + minStay:{ + type:"Int", + array:false, + arrayRequired:false, + required:true + }, + maxStay:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + availability:{ + type:"AvailabilityInput", + array:false, + arrayRequired:false, + required:true + }, + mainImages:{ + type:"MainImage", + array:false, + arrayRequired:false, + required:true + }, + equipmentIds:{ + type:"String", + array:true, + arrayRequired:true, + required:true + }, + featureIds:{ + type:"String", + array:true, + arrayRequired:true, + required:true + }, + address:{ + type:"AddressStrictInput", + array:false, + arrayRequired:false, + required:true + } + }, + SEOSimpleInput:{ + title:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + description:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + keywords:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + thumbnail:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + } + }, + AvailabilityInput:{ + dates:{ + type:"DateRangeComponentInput", + array:false, + arrayRequired:false, + required:true + }, + hours:{ + type:"HoursInput", + array:true, + arrayRequired:false, + required:true + }, + exceptions:{ + type:"DateExceptionComponentInput", + array:true, + arrayRequired:false, + required:true + }, + noWeekend:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + } + }, + HoursInput:{ + from:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + to:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + DateExceptionComponentInput:{ + dates:{ + type:"DateRangeComponentInput", + array:false, + arrayRequired:false, + required:true + }, + hours:{ + type:"HoursInput", + array:true, + arrayRequired:false, + required:true + }, + allDay:{ + type:"Boolean", + array:false, + arrayRequired:false, + required:false + } + }, + MainImage:{ + main:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + one:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + two:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + three:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + four:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + } + }, + EditWorkspaceInput:{ + title:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + teaser:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + cover:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + thumbnail:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + }, + extraImages:{ + type:"ImageInput", + array:true, + arrayRequired:false, + required:true + }, + content:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + seo:{ + type:"SEOSimpleInput", + array:false, + arrayRequired:false, + required:false + }, + urls:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + workspaceType:{ + type:"WorkspaceTypeEnum", + array:false, + arrayRequired:false, + required:false + }, + pricingPerHour:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + currency:{ + type:"AvailableCurrency", + array:false, + arrayRequired:false, + required:true + }, + maxCapacity:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + minStay:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + maxStay:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + availability:{ + type:"AvailabilityInput", + array:false, + arrayRequired:false, + required:false + }, + mainImages:{ + type:"MainImage", + array:false, + arrayRequired:false, + required:true + }, + equipmentIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + featureIds:{ + type:"String", + array:true, + arrayRequired:false, + required:true + }, + address:{ + type:"AddressStrictInput", + array:false, + arrayRequired:false, + required:false + } + }, + CheckoutInput:{ + contactInfo:{ + type:"BuyerInfo", + array:false, + arrayRequired:false, + required:true + }, + billingInfo:{ + type:"BuyerInfo", + array:false, + arrayRequired:false, + required:true + }, + token:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + BuyerInfo:{ + firstName:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + lastName:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + email:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + company:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + vatNumber:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + address:{ + type:"AddressInput", + array:false, + arrayRequired:false, + required:true + } + }, + NewRoomInput:{ + accountIds:{ + type:"String", + array:true, + arrayRequired:true, + required:true + } + }, + AddPeopleToRoomInput:{ + roomId:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + accountIds:{ + type:"String", + array:true, + arrayRequired:true, + required:true + } + }, + RemovePeopleFromRoomInput:{ + roomId:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + accountIds:{ + type:"String", + array:true, + arrayRequired:true, + required:true + } + }, + BookingNewInputSchema:{ + durationType:{ + type:"DurationTypeEnum", + array:false, + arrayRequired:false, + required:true + }, + startToEnd:{ + type:"DateRangeComponentInput", + array:false, + arrayRequired:false, + required:false + }, + slot:{ + type:"SlotComponentInput", + array:false, + arrayRequired:false, + required:false + }, + capacity:{ + type:"Int", + array:false, + arrayRequired:false, + required:false + }, + comments:{ + type:"String", + array:false, + arrayRequired:false, + required:false + }, + ressourceModel:{ + type:"BookingsRessourceEnum", + array:false, + arrayRequired:false, + required:true + }, + ressourceId:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + checkoutInfo:{ + type:"CheckoutEngineInputI", + array:false, + arrayRequired:false, + required:false + } + }, + SlotComponentInput:{ + startDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:true + }, + endDate:{ + type:"DateTime", + array:false, + arrayRequired:false, + required:true + }, + startTime:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + endTime:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + }, + CheckoutEngineInputI:{ + contactInfo:{ + type:"BuyerInfo", + array:false, + arrayRequired:false, + required:true + }, + billingInfo:{ + type:"BuyerInfo", + array:false, + arrayRequired:false, + required:true + }, + token:{ + type:"String", + array:false, + arrayRequired:false, + required:false + } + }, + NewMessageInput:{ + roomId:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + message:{ + type:"String", + array:false, + arrayRequired:false, + required:true + }, + file:{ + type:"ImageInput", + array:false, + arrayRequired:false, + required:false + } + }, + EditMessageInput:{ + message:{ + type:"String", + array:false, + arrayRequired:false, + required:true + } + } +} + +export const ReturnTypes: Record = { + Query:{ + me:"AccountModel", + login:"FirebaseTokenResult", + magicLink:"SimpleResult", + placesAutocomplete:"JSONObject", + placesGeocode:"JSONObject", + placesGeocodeFromAddress:"JSONObject", + categoriesGetOne:"CategoryModel", + categoriesGetMany:"CategoryModel", + categoriesGetCount:"Float", + listsGetOne:"ListModel", + listsGetMany:"ListModel", + listsGetCount:"Float", + accountsDashboardLinkStripe:"String", + workspacesGetOne:"WorkspaceModel", + workspacesGetMany:"WorkspaceModel", + workspacesGetCount:"Float", + workspacesSearchMany:"WorkspaceModel", + workspacesGetMine:"WorkspaceModel", + workspacesGetMineCount:"Float", + bookingsGetOne:"BookingEngineSchema", + bookingsGetMany:"BookingEngineSchema", + bookingsGetCount:"Float", + myCurrentOrder:"OrderEngineSchema", + myOrdersGetOne:"OrderEngineSchema", + myOrdersGetMany:"OrderEngineSchema", + myOrdersGetManyCount:"Float", + roomsGetOne:"RoomModel", + roomsGetMany:"RoomModel", + roomsGetMessages:"MessageModel" + }, + AccountModel:{ + _id:"ID", + organisationId:"ID", + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID", + createdAt:"DateTime", + updatedAt:"DateTime", + r:"String", + w:"String", + d:"String", + types:"AccountTypeEnum", + organisationIds:"String", + email:"String", + userName:"String", + firstName:"String", + lastName:"String", + profilePicture:"ImageComponent", + phoneNumber:"String", + gender:"AccountGenderEnum", + birthDate:"DateTime", + getAge:"Int", + address:"AddressComponent", + language:"AvailableTranslation", + settings:"SettingsComponent", + securityCheck:"Boolean", + paymentInfo:"AccountPaymentInfo", + payoutInfo:"AccountPayoutInfo", + terms:"Boolean", + favLikes:"FavoriteComponent", + getFavorites:"FavoriteComponent" + }, + ImageComponent:{ + title:"String", + fileType:"String", + large:"String", + medium:"String", + small:"String" + }, + AddressComponent:{ + number:"String", + street:"String", + streetBis:"String", + floor:"String", + box:"String", + zip:"String", + state:"String", + city:"String", + country:"String" + }, + SettingsComponent:{ + type:"SettingsTypeEnum", + email:"Boolean", + pushNotifications:"Boolean", + sms:"Boolean" + }, + AccountPaymentInfo:{ + stripeInfo:"StripeInfo", + paymentMethods:"PaymentMethodDetail", + billingInfos:"BillingAddressComponent" + }, + StripeInfo:{ + customerId:"String" + }, + PaymentMethodDetail:{ + _id:"String", + sPayMethodId:"String", + default:"Boolean", + nameOnCard:"String", + type:"cardTypeEnum", + cardInfo:"CardInfo" + }, + CardInfo:{ + brand:"cardBrandEnum", + country:"String", + exp_month:"Int", + exp_year:"Int", + last4:"String" + }, + BillingAddressComponent:{ + firstName:"String", + lastName:"String", + address:"AddressStrictComponent", + email:"String", + default:"Boolean", + customTags:"JSONObject", + company:"String", + vatNumber:"String", + fiscalForm:"String", + _id:"String" + }, + AddressStrictComponent:{ + number:"String", + street:"String", + zip:"String", + city:"String", + country:"String", + streetBis:"String", + floor:"String", + box:"String", + state:"String" + }, + AccountPayoutInfo:{ + stripeInfo:"StripePayoutInfo" + }, + StripePayoutInfo:{ + accountId:"String", + chargesEnabled:"Boolean", + payoutsEnabled:"Boolean", + detailsSubmitted:"Boolean" + }, + FavoriteComponent:{ + ressourceId:"String", + ressourceType:"RessourceEnum", + addedAt:"DateTime", + favorite:"Boolean", + liked:"Boolean" + }, + FirebaseTokenResult:{ + localId:"String", + email:"String", + displayName:"String", + idToken:"String", + registered:"Boolean", + refreshToken:"String", + expiresIn:"String" + }, + SimpleResult:{ + message:"String" + }, + CategoryModel:{ + _id:"ID", + organisationId:"ID", + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID", + createdAt:"DateTime", + updatedAt:"DateTime", + r:"String", + w:"String", + d:"String", + title:"TranslatableComponent", + teaser:"TranslatableComponent", + cover:"ImageComponent", + thumbnail:"ImageComponent", + extraImages:"ImageComponent", + content:"TranslatableComponent", + seo:"SEOField", + urls:"TranslatableComponent", + ressourceType:"RessourceEnum", + colorCode:"String" + }, + TranslatableComponent:{ + en:"String", + fr:"String", + nl:"String", + de:"String" + }, + SEOField:{ + title:"TranslatableComponent", + description:"TranslatableComponent", + keywords:"TranslatableComponent", + thumbnail:"ImageComponent" + }, + ListModel:{ + _id:"ID", + organisationId:"ID", + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID", + createdAt:"DateTime", + updatedAt:"DateTime", + r:"String", + w:"String", + d:"String", + title:"TranslatableComponent", + teaser:"TranslatableComponent", + cover:"ImageComponent", + thumbnail:"ImageComponent", + extraImages:"ImageComponent", + content:"TranslatableComponent", + seo:"SEOField", + urls:"TranslatableComponent", + ressourceType:"ListEnum", + colorCode:"String" + }, + WorkspaceModel:{ + _id:"ID", + organisationId:"ID", + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID", + createdAt:"DateTime", + updatedAt:"DateTime", + r:"String", + w:"String", + d:"String", + title:"String", + teaser:"String", + cover:"ImageComponent", + thumbnail:"ImageComponent", + extraImages:"ImageComponent", + content:"String", + seo:"SEOSimpleField", + urls:"String", + place:"PlaceComponent", + dist:"GeolocDistComponent", + getDistFromLocation:"Float", + workspaceType:"WorkspaceTypeEnum", + pricingPerHour:"Int", + currency:"AvailableCurrency", + maxCapacity:"Int", + minStay:"Int", + maxStay:"Int", + availability:"AvailabilityComponent", + mainImages:"MainImageComponent", + equipmentIds:"String", + featureIds:"String", + address:"AddressComponent", + finalPrice:"Float", + getEquipments:"ListModel", + getFeatures:"ListModel", + getBookings:"BookingSchema", + getOwner:"WorkspaceOwnerComponent" + }, + SEOSimpleField:{ + title:"String", + description:"String", + keywords:"String", + thumbnail:"ImageComponent" + }, + PlaceComponent:{ + placeId:"String", + address:"AddressComponent", + loc:"LocComponent", + formattedAddress:"String" + }, + LocComponent:{ + type:"String", + coordinates:"Float" + }, + GeolocDistComponent:{ + calculated:"Float", + location:"LocComponent" + }, + AvailabilityComponent:{ + dates:"DateRangeComponent", + hours:"HourComponent", + exceptions:"DateExceptionComponent", + noWeekend:"Boolean" + }, + DateRangeComponent:{ + startDate:"DateTime", + endDate:"DateTime" + }, + HourComponent:{ + from:"String", + to:"String" + }, + DateExceptionComponent:{ + dates:"DateRangeComponent", + hours:"HourComponent", + allDay:"Boolean" + }, + MainImageComponent:{ + main:"ImageComponent", + one:"ImageComponent", + two:"ImageComponent", + three:"ImageComponent", + four:"ImageComponent" + }, + BookingSchema:{ + durationType:"DurationTypeEnum", + startToEnd:"DateRangeComponent", + slot:"SlotComponent", + capacity:"Int", + comments:"String", + status:"StatusEnum", + paths:"EnginePathComponent", + ownerId:"String", + dates:"DateRangeComponent", + _id:"ID", + organisationId:"ID", + by:"MetaBy", + permissions:"MetaPermissions", + createdAt:"DateTime", + updatedAt:"DateTime", + tagsIds:"String" + }, + IEngineSchema:{ + "...on BookingSchema": "BookingSchema", + "...on BookingEngineSchema": "BookingEngineSchema", + "...on OrderEngineSchema": "OrderEngineSchema", + _id:"ID", + organisationId:"ID", + paths:"EnginePathComponent", + by:"MetaBy", + permissions:"MetaPermissions", + createdAt:"DateTime", + updatedAt:"DateTime", + tagsIds:"String" + }, + EnginePathComponent:{ + ressourceModel:"ModelLoadersEnum", + ressourceId:"String" + }, + MetaBy:{ + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID" + }, + MetaPermissions:{ + r:"String", + w:"String", + d:"String" + }, + SlotComponent:{ + startDate:"DateTime", + endDate:"DateTime", + startTime:"String", + endTime:"String" + }, + WorkspaceOwnerComponent:{ + firstName:"String", + lastName:"String" + }, + BookingEngineSchema:{ + durationType:"DurationTypeEnum", + startToEnd:"DateRangeComponent", + slot:"SlotComponent", + capacity:"Int", + comments:"String", + status:"StatusEnum", + paths:"EnginePathComponent", + ownerId:"String", + dates:"DateRangeComponent", + owner:"AccountModel", + _id:"ID", + organisationId:"ID", + by:"MetaBy", + permissions:"MetaPermissions", + createdAt:"DateTime", + updatedAt:"DateTime", + tagsIds:"String", + workspace:"WorkspaceModel", + room:"RoomModel" + }, + RoomModel:{ + _id:"ID", + organisationId:"ID", + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID", + createdAt:"DateTime", + updatedAt:"DateTime", + r:"String", + w:"String", + d:"String", + title:"String", + stats:"RoomsStats", + getAccounts:"AccountModel", + getMessages:"MessageModel", + lastMessage:"DateTime" + }, + RoomsStats:{ + unreadCounts:"UnreadCount" + }, + UnreadCount:{ + accountId:"String", + count:"Float" + }, + MessageModel:{ + _id:"ID", + organisationId:"ID", + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID", + createdAt:"DateTime", + updatedAt:"DateTime", + r:"String", + w:"String", + d:"String", + roomId:"String", + notified:"Boolean", + message:"String", + edited:"Boolean", + deleted:"Boolean", + file:"ImageComponent", + sentBy:"String", + readBy:"MessageReadBy" + }, + MessageReadBy:{ + accountId:"String", + readAt:"DateTime" + }, + OrderEngineSchema:{ + lines:"LineItemSchema", + localeInfo:"LocaleInfo", + currency:"CurrencyEnum", + vatExempt:"Boolean", + accountId:"String", + contactInfo:"BuyerInfoComponent", + billingInfo:"BuyerInfoComponent", + promoId:"ID", + invoiceInfo:"InvoiceInfo", + providerOrderItems:"ProviderSchema", + orderName:"TranslatableComponent", + orderStatus:"OrderStatusEnum", + paymentIntent:"PaymentIntentInfo", + paymentStatus:"PaymentStatusEnum", + paymentInfo:"PaymentInfo", + subTotalPrice:"Float", + vatPrice:"Float", + finalPrice:"Float", + totalPrice:"Float", + promo:"PromoModel", + _id:"ID", + organisationId:"ID", + paths:"EnginePathComponent", + by:"MetaBy", + permissions:"MetaPermissions", + createdAt:"DateTime", + updatedAt:"DateTime", + tagsIds:"String", + getBookingInput:"BookingNewInput" + }, + LineItemSchema:{ + _id:"String", + productRessource:"ProductRessourceEnum", + productId:"String", + title:"String", + price:"Float", + salesPrice:"Float", + quantity:"Float", + vatClassId:"String", + parentId:"String", + addOns:"AddOnComponent", + localeInfo:"LocaleInfo", + promoId:"String", + finalPrice:"Float", + getProduct:"ProductUnion", + getLinePrice:"Float", + getLineVatPrice:"Float" + }, + AddOnComponent:{ + addOnId:"String", + quantity:"Float" + }, + LocaleInfo:{ + countryCode:"String", + locale:"AvailableTranslation" + }, + ProductUnion:{ + "...on WorkspaceModel":"WorkspaceModel" + }, + BuyerInfoComponent:{ + firstName:"String", + lastName:"String", + email:"String", + company:"String", + vatNumber:"String", + address:"AddressComponent" + }, + InvoiceInfo:{ + provider:"InvoicingProvider", + invoiceId:"String", + invoiceNumber:"Float" + }, + ProviderSchema:{ + _id:"String", + organisationId:"String", + orderStatus:"OrderBusinessStatusEnum", + lines:"LineItemSchema" + }, + PaymentIntentInfo:{ + provider:"PaymentProvider", + stripePaymentIntentData:"StripePaymentIntentData" + }, + StripePaymentIntentData:{ + id:"String", + client_secret:"String", + currency:"String", + customer:"String", + status:"String" + }, + PaymentInfo:{ + provider:"PaymentProvider", + transactionId:"String" + }, + PromoModel:{ + _id:"ID", + organisationId:"ID", + createdBy:"ID", + updatedBy:"ID", + deletedBy:"ID", + createdAt:"DateTime", + updatedAt:"DateTime", + r:"String", + w:"String", + d:"String", + code:"String", + description:"TranslatableComponent", + type:"PromoType", + value:"Float", + validity:"DateTime", + cummulable:"Boolean", + usageLimit:"Float", + usage:"Float" + }, + BookingNewInput:{ + durationType:"DurationTypeEnum", + startToEnd:"DateRangeComponent", + slot:"SlotComponent", + capacity:"Int", + comments:"String", + ressourceModel:"BookingsRessourceEnum", + ressourceId:"String", + checkoutInfo:"CheckoutEngineInput" + }, + CheckoutEngineInput:{ + contactInfo:"BuyerInfoComponent", + billingInfo:"BuyerInfoComponent", + token:"String" + }, + Mutation:{ + updateMe:"AccountModel", + updateMeEmail:"AccountModel", + updateMePassword:"AccountModel", + resetPassword:"SimpleResult", + registerGuest:"AccountModel", + register:"AccountModel", + accountsBillingInfosAddOne:"BillingAddressComponent", + accountsBillingInfosEditOne:"BillingAddressComponent", + accountsBillingInfosDeleteOne:"String", + accountsPaymentMethodsAddOne:"PaymentMethodDetail", + accountsPaymentMethodsMarkasDefault:"PaymentMethodDetail", + accountsPaymentMethodsDeleteOne:"String", + accountsLinkWithStripe:"String", + accountsRefreshInfoStripe:"AccountModel", + accountsFavoritesAddOne:"FavoriteComponent", + accountsFavoritesRemoveOne:"SuccessResponse", + workspacesAddOne:"WorkspaceModel", + workspacesEditOne:"WorkspaceModel", + workspacesDeleteOne:"WorkspaceModel", + finaliseCheckoutWithStripe:"OrderEngineSchema", + removeCurrentOrder:"OrderEngineSchema", + validateOrder:"OrderEngineSchema", + roomsCreateOne:"RoomModel", + roomsAddPeople:"RoomModel", + roomsRemovePeople:"RoomModel", + roomsMarkAllMessageAsRead:"RoomModel", + reserveOneBooking:"OrderEngineSchema", + roomsAddOneMessage:"MessageModel", + roomsEditOneMessage:"MessageModel", + roomsMarkMessageAsRead:"MessageModel", + roomsDeleteOneMessage:"MessageModel" + }, + SuccessResponse:{ + success:"Boolean" + } +} \ No newline at end of file diff --git a/__tests/app/helpers/zeus/index.ts b/__tests/app/helpers/zeus/index.ts new file mode 100644 index 0000000..889f11b --- /dev/null +++ b/__tests/app/helpers/zeus/index.ts @@ -0,0 +1,3420 @@ +/* eslint-disable */ + +import { AllTypesProps, ReturnTypes } from './const'; +type ZEUS_INTERFACES = GraphQLTypes["IEngineSchema"] +type ZEUS_UNIONS = GraphQLTypes["ProductUnion"] + +export type ValueTypes = { + ["Query"]: AliasType<{ + me?:ValueTypes["AccountModel"], +login?: [{ refreshToken?:string, creds?:ValueTypes["LoginInput"]},ValueTypes["FirebaseTokenResult"]], +magicLink?: [{ input:ValueTypes["ResetPasswordInput"]},ValueTypes["SimpleResult"]], +placesAutocomplete?: [{ session:string, input:string, language?:ValueTypes["AvailableTranslation"], country?:ValueTypes["CountryCodesComponentEnum"][]},true], +placesGeocode?: [{ session:string, placeId:string},true], +placesGeocodeFromAddress?: [{ address:string},true], +categoriesGetOne?: [{ id:string},ValueTypes["CategoryModel"]], +categoriesGetMany?: [{ search?:string, ressourceType:ValueTypes["RessourceEnum"], pagination?:ValueTypes["GetArgs"]},ValueTypes["CategoryModel"]], +categoriesGetCount?: [{ search?:string, ressourceType:ValueTypes["RessourceEnum"], pagination?:ValueTypes["GetArgs"]},true], +listsGetOne?: [{ id:string},ValueTypes["ListModel"]], +listsGetMany?: [{ search?:string, ressourceType:ValueTypes["ListEnum"], pagination?:ValueTypes["GetArgs"]},ValueTypes["ListModel"]], +listsGetCount?: [{ search?:string, ressourceType:ValueTypes["ListEnum"], pagination?:ValueTypes["GetArgs"]},true], + accountsDashboardLinkStripe?:true, +workspacesGetOne?: [{ id:string},ValueTypes["WorkspaceModel"]], +workspacesGetMany?: [{ geoSearch?:ValueTypes["GeolocSearchInput"], geoSearchWithAddress?:ValueTypes["GeolocAddressSearchInput"], _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"], dateFrom?:ValueTypes["DateTime"], dateTo?:ValueTypes["DateTime"], hoursFrom?:string, hoursTo?:string, workspaceType?:ValueTypes["WorkspaceTypeEnum"], workspaceTypes?:ValueTypes["WorkspaceTypeEnum"][], priceMin?:number, priceMax?:number, maxCapacity?:number, equipmentIds?:string[], featureIds?:string[]},ValueTypes["WorkspaceModel"]], +workspacesGetCount?: [{ geoSearch?:ValueTypes["GeolocSearchInput"], geoSearchWithAddress?:ValueTypes["GeolocAddressSearchInput"], _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"], dateFrom?:ValueTypes["DateTime"], dateTo?:ValueTypes["DateTime"], hoursFrom?:string, hoursTo?:string, workspaceType?:ValueTypes["WorkspaceTypeEnum"], workspaceTypes?:ValueTypes["WorkspaceTypeEnum"][], priceMin?:number, priceMax?:number, maxCapacity?:number, equipmentIds?:string[], featureIds?:string[]},true], +workspacesSearchMany?: [{ geoSearch?:ValueTypes["GeolocSearchInput"], geoSearchWithAddress?:ValueTypes["GeolocAddressSearchInput"], _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"], dateFrom?:ValueTypes["DateTime"], dateTo?:ValueTypes["DateTime"], hoursFrom?:string, hoursTo?:string, workspaceType?:ValueTypes["WorkspaceTypeEnum"], workspaceTypes?:ValueTypes["WorkspaceTypeEnum"][], priceMin?:number, priceMax?:number, maxCapacity?:number, equipmentIds?:string[], featureIds?:string[]},ValueTypes["WorkspaceModel"]], +workspacesGetMine?: [{ geoSearch?:ValueTypes["GeolocSearchInput"], geoSearchWithAddress?:ValueTypes["GeolocAddressSearchInput"], _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"], dateFrom?:ValueTypes["DateTime"], dateTo?:ValueTypes["DateTime"], hoursFrom?:string, hoursTo?:string, workspaceType?:ValueTypes["WorkspaceTypeEnum"], workspaceTypes?:ValueTypes["WorkspaceTypeEnum"][], priceMin?:number, priceMax?:number, maxCapacity?:number, equipmentIds?:string[], featureIds?:string[]},ValueTypes["WorkspaceModel"]], +workspacesGetMineCount?: [{ geoSearch?:ValueTypes["GeolocSearchInput"], geoSearchWithAddress?:ValueTypes["GeolocAddressSearchInput"], _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"], dateFrom?:ValueTypes["DateTime"], dateTo?:ValueTypes["DateTime"], hoursFrom?:string, hoursTo?:string, workspaceType?:ValueTypes["WorkspaceTypeEnum"], workspaceTypes?:ValueTypes["WorkspaceTypeEnum"][], priceMin?:number, priceMax?:number, maxCapacity?:number, equipmentIds?:string[], featureIds?:string[]},true], +bookingsGetOne?: [{ id:string},ValueTypes["BookingEngineSchema"]], +bookingsGetMany?: [{ startDate?:ValueTypes["DateTime"], endDate?:ValueTypes["DateTime"], rangeType?:ValueTypes["RangeType"], dateTabType?:ValueTypes["DateTabType"], dateRange?:ValueTypes["DateRangeComponentInput"], _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"], ressourceModel?:ValueTypes["BookingsRessourceEnum"], ressourceId?:string, status?:ValueTypes["StatusEnum"][]},ValueTypes["BookingEngineSchema"]], +bookingsGetCount?: [{ startDate?:ValueTypes["DateTime"], endDate?:ValueTypes["DateTime"], rangeType?:ValueTypes["RangeType"], dateTabType?:ValueTypes["DateTabType"], dateRange?:ValueTypes["DateRangeComponentInput"], _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"], ressourceModel?:ValueTypes["BookingsRessourceEnum"], ressourceId?:string, status?:ValueTypes["StatusEnum"][]},true], + myCurrentOrder?:ValueTypes["OrderEngineSchema"], +myOrdersGetOne?: [{ id:string},ValueTypes["OrderEngineSchema"]], +myOrdersGetMany?: [{ orderStatus?:ValueTypes["OrderStatusEnum"], paymentStatus?:ValueTypes["PaymentStatusEnum"], pagination?:ValueTypes["GetArgs"], search?:string},ValueTypes["OrderEngineSchema"]], +myOrdersGetManyCount?: [{ orderStatus?:ValueTypes["OrderStatusEnum"], paymentStatus?:ValueTypes["PaymentStatusEnum"], pagination?:ValueTypes["GetArgs"], search?:string},true], +roomsGetOne?: [{ id:string},ValueTypes["RoomModel"]], +roomsGetMany?: [{ pagination?:ValueTypes["GetArgs"]},ValueTypes["RoomModel"]], +roomsGetMessages?: [{ pagination?:ValueTypes["GetArgs"], id:string},ValueTypes["MessageModel"]], + __typename?: true +}>; + ["AccountModel"]: AliasType<{ + _id?:true, + organisationId?:true, + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + createdAt?:true, + updatedAt?:true, + r?:true, + w?:true, + d?:true, + types?:true, + organisationIds?:true, + email?:true, + userName?:true, + firstName?:true, + lastName?:true, + profilePicture?:ValueTypes["ImageComponent"], + phoneNumber?:true, + gender?:true, + birthDate?:true, + getAge?:true, + address?:ValueTypes["AddressComponent"], + language?:true, + settings?:ValueTypes["SettingsComponent"], + securityCheck?:true, + paymentInfo?:ValueTypes["AccountPaymentInfo"], + payoutInfo?:ValueTypes["AccountPayoutInfo"], + terms?:true, + favLikes?:ValueTypes["FavoriteComponent"], + getFavorites?:ValueTypes["FavoriteComponent"], + __typename?: true +}>; + /** The javascript `Date` as string. Type represents date and time as the ISO Date string. */ +["DateTime"]:unknown; + ["AccountTypeEnum"]:AccountTypeEnum; + ["ImageComponent"]: AliasType<{ + title?:true, + fileType?:true, + large?:true, + medium?:true, + small?:true, + __typename?: true +}>; + ["AccountGenderEnum"]:AccountGenderEnum; + ["AddressComponent"]: AliasType<{ + number?:true, + street?:true, + streetBis?:true, + floor?:true, + box?:true, + zip?:true, + state?:true, + city?:true, + country?:true, + __typename?: true +}>; + ["AvailableTranslation"]:AvailableTranslation; + ["SettingsComponent"]: AliasType<{ + type?:true, + email?:true, + pushNotifications?:true, + sms?:true, + __typename?: true +}>; + ["SettingsTypeEnum"]:SettingsTypeEnum; + ["AccountPaymentInfo"]: AliasType<{ + stripeInfo?:ValueTypes["StripeInfo"], + paymentMethods?:ValueTypes["PaymentMethodDetail"], + billingInfos?:ValueTypes["BillingAddressComponent"], + __typename?: true +}>; + ["StripeInfo"]: AliasType<{ + customerId?:true, + __typename?: true +}>; + ["PaymentMethodDetail"]: AliasType<{ + _id?:true, + sPayMethodId?:true, + default?:true, + nameOnCard?:true, + type?:true, + cardInfo?:ValueTypes["CardInfo"], + __typename?: true +}>; + ["cardTypeEnum"]:cardTypeEnum; + ["CardInfo"]: AliasType<{ + brand?:true, + country?:true, + exp_month?:true, + exp_year?:true, + last4?:true, + __typename?: true +}>; + ["cardBrandEnum"]:cardBrandEnum; + ["BillingAddressComponent"]: AliasType<{ + firstName?:true, + lastName?:true, + address?:ValueTypes["AddressStrictComponent"], + email?:true, + default?:true, + customTags?:true, + company?:true, + vatNumber?:true, + fiscalForm?:true, + _id?:true, + __typename?: true +}>; + ["AddressStrictComponent"]: AliasType<{ + number?:true, + street?:true, + zip?:true, + city?:true, + country?:true, + streetBis?:true, + floor?:true, + box?:true, + state?:true, + __typename?: true +}>; + /** The `JSONObject` scalar type represents JSON objects as specified by [ECMA-404](http://www.ecma-international.org/publications/files/ECMA-ST/ECMA-404.pdf). */ +["JSONObject"]:unknown; + ["AccountPayoutInfo"]: AliasType<{ + stripeInfo?:ValueTypes["StripePayoutInfo"], + __typename?: true +}>; + ["StripePayoutInfo"]: AliasType<{ + accountId?:true, + chargesEnabled?:true, + payoutsEnabled?:true, + detailsSubmitted?:true, + __typename?: true +}>; + ["FavoriteComponent"]: AliasType<{ + ressourceId?:true, + ressourceType?:true, + addedAt?:true, + favorite?:true, + liked?:true, + __typename?: true +}>; + ["RessourceEnum"]:RessourceEnum; + ["FirebaseTokenResult"]: AliasType<{ + localId?:true, + email?:true, + displayName?:true, + idToken?:true, + registered?:true, + refreshToken?:true, + expiresIn?:true, + __typename?: true +}>; + ["LoginInput"]: { + email:string, + password:string +}; + ["SimpleResult"]: AliasType<{ + message?:true, + __typename?: true +}>; + ["ResetPasswordInput"]: { + email:string +}; + ["CountryCodesComponentEnum"]:CountryCodesComponentEnum; + ["CategoryModel"]: AliasType<{ + _id?:true, + organisationId?:true, + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + createdAt?:true, + updatedAt?:true, + r?:true, + w?:true, + d?:true, + title?:ValueTypes["TranslatableComponent"], + teaser?:ValueTypes["TranslatableComponent"], + cover?:ValueTypes["ImageComponent"], + thumbnail?:ValueTypes["ImageComponent"], + extraImages?:ValueTypes["ImageComponent"], + content?:ValueTypes["TranslatableComponent"], + seo?:ValueTypes["SEOField"], + urls?:ValueTypes["TranslatableComponent"], + ressourceType?:true, + colorCode?:true, + __typename?: true +}>; + ["TranslatableComponent"]: AliasType<{ + en?:true, + fr?:true, + nl?:true, + de?:true, + __typename?: true +}>; + ["SEOField"]: AliasType<{ + title?:ValueTypes["TranslatableComponent"], + description?:ValueTypes["TranslatableComponent"], + keywords?:ValueTypes["TranslatableComponent"], + thumbnail?:ValueTypes["ImageComponent"], + __typename?: true +}>; + ["GetArgs"]: { + limit:number, + skip:number, + sort?:string +}; + ["ListModel"]: AliasType<{ + _id?:true, + organisationId?:true, + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + createdAt?:true, + updatedAt?:true, + r?:true, + w?:true, + d?:true, + title?:ValueTypes["TranslatableComponent"], + teaser?:ValueTypes["TranslatableComponent"], + cover?:ValueTypes["ImageComponent"], + thumbnail?:ValueTypes["ImageComponent"], + extraImages?:ValueTypes["ImageComponent"], + content?:ValueTypes["TranslatableComponent"], + seo?:ValueTypes["SEOField"], + urls?:ValueTypes["TranslatableComponent"], + ressourceType?:true, + colorCode?:true, + __typename?: true +}>; + ["ListEnum"]:ListEnum; + ["WorkspaceModel"]: AliasType<{ + _id?:true, + organisationId?:true, + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + createdAt?:true, + updatedAt?:true, + r?:true, + w?:true, + d?:true, + title?:true, + teaser?:true, + cover?:ValueTypes["ImageComponent"], + thumbnail?:ValueTypes["ImageComponent"], + extraImages?:ValueTypes["ImageComponent"], + content?:true, + seo?:ValueTypes["SEOSimpleField"], + urls?:true, + place?:ValueTypes["PlaceComponent"], + dist?:ValueTypes["GeolocDistComponent"], +getDistFromLocation?: [{ longitude:number, latitude:number, radius?:number},true], + workspaceType?:true, + pricingPerHour?:true, + currency?:true, + maxCapacity?:true, + minStay?:true, + maxStay?:true, + availability?:ValueTypes["AvailabilityComponent"], + mainImages?:ValueTypes["MainImageComponent"], + equipmentIds?:true, + featureIds?:true, + address?:ValueTypes["AddressComponent"], + finalPrice?:true, + getEquipments?:ValueTypes["ListModel"], + getFeatures?:ValueTypes["ListModel"], +getBookings?: [{ ressourceModel?:ValueTypes["BookingsRessourceEnum"], ressourceId?:string, status?:ValueTypes["StatusEnum"][], startDate?:ValueTypes["DateTime"], endDate?:ValueTypes["DateTime"], rangeType?:ValueTypes["RangeType"], dateTabType?:ValueTypes["DateTabType"], dateRange?:ValueTypes["DateRangeComponentInput"], _ids?:string[], search?:string, afterCreatedAt?:ValueTypes["DateTime"], afterUpdatedAt?:ValueTypes["DateTime"], pagination?:ValueTypes["GetArgs"]},ValueTypes["BookingSchema"]], + getOwner?:ValueTypes["WorkspaceOwnerComponent"], + __typename?: true +}>; + ["SEOSimpleField"]: AliasType<{ + title?:true, + description?:true, + keywords?:true, + thumbnail?:ValueTypes["ImageComponent"], + __typename?: true +}>; + ["PlaceComponent"]: AliasType<{ + placeId?:true, + address?:ValueTypes["AddressComponent"], + loc?:ValueTypes["LocComponent"], + formattedAddress?:true, + __typename?: true +}>; + ["LocComponent"]: AliasType<{ + type?:true, + coordinates?:true, + __typename?: true +}>; + ["GeolocDistComponent"]: AliasType<{ + calculated?:true, + location?:ValueTypes["LocComponent"], + __typename?: true +}>; + ["WorkspaceTypeEnum"]:WorkspaceTypeEnum; + ["AvailableCurrency"]:AvailableCurrency; + ["AvailabilityComponent"]: AliasType<{ + dates?:ValueTypes["DateRangeComponent"], + hours?:ValueTypes["HourComponent"], + exceptions?:ValueTypes["DateExceptionComponent"], + noWeekend?:true, + __typename?: true +}>; + ["DateRangeComponent"]: AliasType<{ + startDate?:true, + endDate?:true, + __typename?: true +}>; + ["HourComponent"]: AliasType<{ + from?:true, + to?:true, + __typename?: true +}>; + ["DateExceptionComponent"]: AliasType<{ + dates?:ValueTypes["DateRangeComponent"], + hours?:ValueTypes["HourComponent"], + allDay?:true, + __typename?: true +}>; + ["MainImageComponent"]: AliasType<{ + main?:ValueTypes["ImageComponent"], + one?:ValueTypes["ImageComponent"], + two?:ValueTypes["ImageComponent"], + three?:ValueTypes["ImageComponent"], + four?:ValueTypes["ImageComponent"], + __typename?: true +}>; + ["BookingSchema"]: AliasType<{ + durationType?:true, + startToEnd?:ValueTypes["DateRangeComponent"], + slot?:ValueTypes["SlotComponent"], + capacity?:true, + comments?:true, + status?:true, + paths?:ValueTypes["EnginePathComponent"], + ownerId?:true, + dates?:ValueTypes["DateRangeComponent"], + _id?:true, + organisationId?:true, + by?:ValueTypes["MetaBy"], + permissions?:ValueTypes["MetaPermissions"], + createdAt?:true, + updatedAt?:true, + tagsIds?:true, + __typename?: true +}>; + ["IEngineSchema"]:AliasType<{ + _id?:true, + organisationId?:true, + paths?:ValueTypes["EnginePathComponent"], + by?:ValueTypes["MetaBy"], + permissions?:ValueTypes["MetaPermissions"], + createdAt?:true, + updatedAt?:true, + tagsIds?:true; + ['...on BookingSchema']?: Omit; + ['...on BookingEngineSchema']?: Omit; + ['...on OrderEngineSchema']?: Omit; + __typename?: true +}>; + ["EnginePathComponent"]: AliasType<{ + ressourceModel?:true, + ressourceId?:true, + __typename?: true +}>; + ["ModelLoadersEnum"]:ModelLoadersEnum; + ["MetaBy"]: AliasType<{ + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + __typename?: true +}>; + ["MetaPermissions"]: AliasType<{ + r?:true, + w?:true, + d?:true, + __typename?: true +}>; + ["DurationTypeEnum"]:DurationTypeEnum; + ["SlotComponent"]: AliasType<{ + startDate?:true, + endDate?:true, + startTime?:true, + endTime?:true, + __typename?: true +}>; + ["StatusEnum"]:StatusEnum; + ["BookingsRessourceEnum"]:BookingsRessourceEnum; + ["RangeType"]:RangeType; + ["DateTabType"]:DateTabType; + ["DateRangeComponentInput"]: { + startDate:ValueTypes["DateTime"], + endDate:ValueTypes["DateTime"] +}; + ["WorkspaceOwnerComponent"]: AliasType<{ + firstName?:true, + lastName?:true, + __typename?: true +}>; + ["GeolocSearchInput"]: { + longitude:number, + latitude:number, + radius?:number +}; + ["GeolocAddressSearchInput"]: { + formattedAddress:string, + radius?:number +}; + ["BookingEngineSchema"]: AliasType<{ + durationType?:true, + startToEnd?:ValueTypes["DateRangeComponent"], + slot?:ValueTypes["SlotComponent"], + capacity?:true, + comments?:true, + status?:true, + paths?:ValueTypes["EnginePathComponent"], + ownerId?:true, + dates?:ValueTypes["DateRangeComponent"], + owner?:ValueTypes["AccountModel"], + _id?:true, + organisationId?:true, + by?:ValueTypes["MetaBy"], + permissions?:ValueTypes["MetaPermissions"], + createdAt?:true, + updatedAt?:true, + tagsIds?:true, + workspace?:ValueTypes["WorkspaceModel"], + room?:ValueTypes["RoomModel"], + __typename?: true +}>; + ["RoomModel"]: AliasType<{ + _id?:true, + organisationId?:true, + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + createdAt?:true, + updatedAt?:true, + r?:true, + w?:true, + d?:true, + title?:true, + stats?:ValueTypes["RoomsStats"], + getAccounts?:ValueTypes["AccountModel"], +getMessages?: [{ pagination?:ValueTypes["GetArgs"]},ValueTypes["MessageModel"]], + lastMessage?:true, + __typename?: true +}>; + ["RoomsStats"]: AliasType<{ + unreadCounts?:ValueTypes["UnreadCount"], + __typename?: true +}>; + ["UnreadCount"]: AliasType<{ + accountId?:true, + count?:true, + __typename?: true +}>; + ["MessageModel"]: AliasType<{ + _id?:true, + organisationId?:true, + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + createdAt?:true, + updatedAt?:true, + r?:true, + w?:true, + d?:true, + roomId?:true, + notified?:true, + message?:true, + edited?:true, + deleted?:true, + file?:ValueTypes["ImageComponent"], + sentBy?:true, + readBy?:ValueTypes["MessageReadBy"], + __typename?: true +}>; + ["MessageReadBy"]: AliasType<{ + accountId?:true, + readAt?:true, + __typename?: true +}>; + ["OrderEngineSchema"]: AliasType<{ + lines?:ValueTypes["LineItemSchema"], + localeInfo?:ValueTypes["LocaleInfo"], + currency?:true, + vatExempt?:true, + accountId?:true, + contactInfo?:ValueTypes["BuyerInfoComponent"], + billingInfo?:ValueTypes["BuyerInfoComponent"], + promoId?:true, + invoiceInfo?:ValueTypes["InvoiceInfo"], + providerOrderItems?:ValueTypes["ProviderSchema"], + orderName?:ValueTypes["TranslatableComponent"], + orderStatus?:true, + paymentIntent?:ValueTypes["PaymentIntentInfo"], + paymentStatus?:true, + paymentInfo?:ValueTypes["PaymentInfo"], + subTotalPrice?:true, + vatPrice?:true, + finalPrice?:true, + totalPrice?:true, + promo?:ValueTypes["PromoModel"], + _id?:true, + organisationId?:true, + paths?:ValueTypes["EnginePathComponent"], + by?:ValueTypes["MetaBy"], + permissions?:ValueTypes["MetaPermissions"], + createdAt?:true, + updatedAt?:true, + tagsIds?:true, + getBookingInput?:ValueTypes["BookingNewInput"], + __typename?: true +}>; + ["LineItemSchema"]: AliasType<{ + _id?:true, + productRessource?:true, + productId?:true, + title?:true, + price?:true, + salesPrice?:true, + quantity?:true, + vatClassId?:true, + parentId?:true, + addOns?:ValueTypes["AddOnComponent"], + localeInfo?:ValueTypes["LocaleInfo"], + promoId?:true, + finalPrice?:true, + getProduct?:ValueTypes["ProductUnion"], + getLinePrice?:true, + getLineVatPrice?:true, + __typename?: true +}>; + ["ProductRessourceEnum"]:ProductRessourceEnum; + ["AddOnComponent"]: AliasType<{ + addOnId?:true, + quantity?:true, + __typename?: true +}>; + ["LocaleInfo"]: AliasType<{ + countryCode?:true, + locale?:true, + __typename?: true +}>; + ["ProductUnion"]: AliasType<{ ["...on WorkspaceModel"] : ValueTypes["WorkspaceModel"] + __typename?: true +}>; + ["CurrencyEnum"]:CurrencyEnum; + ["BuyerInfoComponent"]: AliasType<{ + firstName?:true, + lastName?:true, + email?:true, + company?:true, + vatNumber?:true, + address?:ValueTypes["AddressComponent"], + __typename?: true +}>; + ["InvoiceInfo"]: AliasType<{ + provider?:true, + invoiceId?:true, + invoiceNumber?:true, + __typename?: true +}>; + ["InvoicingProvider"]:InvoicingProvider; + ["ProviderSchema"]: AliasType<{ + _id?:true, + organisationId?:true, + orderStatus?:true, + lines?:ValueTypes["LineItemSchema"], + __typename?: true +}>; + ["OrderBusinessStatusEnum"]:OrderBusinessStatusEnum; + ["OrderStatusEnum"]:OrderStatusEnum; + ["PaymentIntentInfo"]: AliasType<{ + provider?:true, + stripePaymentIntentData?:ValueTypes["StripePaymentIntentData"], + __typename?: true +}>; + ["PaymentProvider"]:PaymentProvider; + ["StripePaymentIntentData"]: AliasType<{ + id?:true, + client_secret?:true, + currency?:true, + customer?:true, + status?:true, + __typename?: true +}>; + ["PaymentStatusEnum"]:PaymentStatusEnum; + ["PaymentInfo"]: AliasType<{ + provider?:true, + transactionId?:true, + __typename?: true +}>; + ["PromoModel"]: AliasType<{ + _id?:true, + organisationId?:true, + createdBy?:true, + updatedBy?:true, + deletedBy?:true, + createdAt?:true, + updatedAt?:true, + r?:true, + w?:true, + d?:true, + code?:true, + description?:ValueTypes["TranslatableComponent"], + type?:true, + value?:true, + validity?:true, + cummulable?:true, + usageLimit?:true, + usage?:true, + __typename?: true +}>; + ["PromoType"]:PromoType; + ["BookingNewInput"]: AliasType<{ + durationType?:true, + startToEnd?:ValueTypes["DateRangeComponent"], + slot?:ValueTypes["SlotComponent"], + capacity?:true, + comments?:true, + ressourceModel?:true, + ressourceId?:true, + checkoutInfo?:ValueTypes["CheckoutEngineInput"], + __typename?: true +}>; + ["CheckoutEngineInput"]: AliasType<{ + contactInfo?:ValueTypes["BuyerInfoComponent"], + billingInfo?:ValueTypes["BuyerInfoComponent"], + token?:true, + __typename?: true +}>; + ["Mutation"]: AliasType<{ +updateMe?: [{ input:ValueTypes["EditAccountInput"]},ValueTypes["AccountModel"]], +updateMeEmail?: [{ input:ValueTypes["NewEmailInput"]},ValueTypes["AccountModel"]], +updateMePassword?: [{ input:ValueTypes["NewPasswordInput"]},ValueTypes["AccountModel"]], +resetPassword?: [{ input:ValueTypes["ResetPasswordInput"]},ValueTypes["SimpleResult"]], +registerGuest?: [{ otherInfo:ValueTypes["EditAccountInput"], input:ValueTypes["LinkEmailInput"]},ValueTypes["AccountModel"]], +register?: [{ input:ValueTypes["NewAccountInput"]},ValueTypes["AccountModel"]], +accountsBillingInfosAddOne?: [{ input:ValueTypes["BillingAddressInput"]},ValueTypes["BillingAddressComponent"]], +accountsBillingInfosEditOne?: [{ input:ValueTypes["BillingAddressInput"], id:string},ValueTypes["BillingAddressComponent"]], +accountsBillingInfosDeleteOne?: [{ id:string},true], +accountsPaymentMethodsAddOne?: [{ input:ValueTypes["PaymentMethodInput"]},ValueTypes["PaymentMethodDetail"]], +accountsPaymentMethodsMarkasDefault?: [{ id:string},ValueTypes["PaymentMethodDetail"]], +accountsPaymentMethodsDeleteOne?: [{ id:string},true], + accountsLinkWithStripe?:true, + accountsRefreshInfoStripe?:ValueTypes["AccountModel"], +accountsFavoritesAddOne?: [{ input:ValueTypes["AddInput"]},ValueTypes["FavoriteComponent"]], +accountsFavoritesRemoveOne?: [{ input:ValueTypes["RemoveInput"]},ValueTypes["SuccessResponse"]], +workspacesAddOne?: [{ input:ValueTypes["NewWorkspaceInput"]},ValueTypes["WorkspaceModel"]], +workspacesEditOne?: [{ input:ValueTypes["EditWorkspaceInput"], id:string},ValueTypes["WorkspaceModel"]], +workspacesDeleteOne?: [{ id:string},ValueTypes["WorkspaceModel"]], +finaliseCheckoutWithStripe?: [{ input:ValueTypes["CheckoutInput"]},ValueTypes["OrderEngineSchema"]], + removeCurrentOrder?:ValueTypes["OrderEngineSchema"], +validateOrder?: [{ id:string},ValueTypes["OrderEngineSchema"]], +roomsCreateOne?: [{ input:ValueTypes["NewRoomInput"]},ValueTypes["RoomModel"]], +roomsAddPeople?: [{ input:ValueTypes["AddPeopleToRoomInput"]},ValueTypes["RoomModel"]], +roomsRemovePeople?: [{ input:ValueTypes["RemovePeopleFromRoomInput"]},ValueTypes["RoomModel"]], +roomsMarkAllMessageAsRead?: [{ roomId:string},ValueTypes["RoomModel"]], +reserveOneBooking?: [{ input:ValueTypes["BookingNewInputSchema"]},ValueTypes["OrderEngineSchema"]], +roomsAddOneMessage?: [{ input:ValueTypes["NewMessageInput"]},ValueTypes["MessageModel"]], +roomsEditOneMessage?: [{ input:ValueTypes["EditMessageInput"], id:string},ValueTypes["MessageModel"]], +roomsMarkMessageAsRead?: [{ id:string},ValueTypes["MessageModel"]], +roomsDeleteOneMessage?: [{ id:string},ValueTypes["MessageModel"]], + __typename?: true +}>; + ["EditAccountInput"]: { + userName?:string, + firstName?:string, + lastName?:string, + profilePicture?:ValueTypes["ImageInput"], + phoneNumber?:string, + gender?:ValueTypes["AccountGenderEnum"], + birthDate?:ValueTypes["DateTime"], + address?:ValueTypes["AddressInput"], + language?:ValueTypes["AvailableTranslation"], + settings?:ValueTypes["SettingsInput"][] +}; + ["ImageInput"]: { + title?:string, + fileType?:string, + large:string, + medium?:string, + small?:string +}; + ["AddressInput"]: { + number?:string, + street?:string, + streetBis?:string, + floor?:string, + box?:string, + zip?:string, + state?:string, + city:string, + country:string +}; + ["SettingsInput"]: { + type:ValueTypes["SettingsTypeEnum"], + email?:boolean, + pushNotifications?:boolean, + sms?:boolean +}; + ["NewEmailInput"]: { + newEmail:string +}; + ["NewPasswordInput"]: { + oldPassword:string, + newPassword:string, + newPasswordConfirmation:string +}; + ["LinkEmailInput"]: { + email:string, + password:string, + idToken:string +}; + ["NewAccountInput"]: { + email:string, + password:string, + passwordConfirmation:string, + userName?:string, + firstName?:string, + lastName?:string, + profilePicture?:ValueTypes["ImageInput"], + phoneNumber?:string, + gender?:ValueTypes["AccountGenderEnum"], + birthDate?:ValueTypes["DateTime"], + address?:ValueTypes["AddressInput"], + language?:ValueTypes["AvailableTranslation"], + settings?:ValueTypes["SettingsInput"][], + terms:boolean +}; + ["BillingAddressInput"]: { + firstName:string, + lastName:string, + address:ValueTypes["AddressStrictInput"], + email?:string, + default?:boolean, + customTags?:ValueTypes["JSONObject"], + company?:string, + vatNumber?:string, + fiscalForm?:string +}; + ["AddressStrictInput"]: { + number:string, + street:string, + zip:string, + city:string, + country:string, + streetBis?:string, + floor?:string, + box?:string, + state?:string +}; + ["PaymentMethodInput"]: { + sPayMethodId:string, + nameOnCard:string, + default?:boolean +}; + ["AddInput"]: { + ressourceId:string, + ressourceType:ValueTypes["RessourceEnum"] +}; + ["SuccessResponse"]: AliasType<{ + success?:true, + __typename?: true +}>; + ["RemoveInput"]: { + ressourceId:string +}; + ["NewWorkspaceInput"]: { + title?:string, + teaser?:string, + cover?:ValueTypes["ImageInput"], + thumbnail?:ValueTypes["ImageInput"], + extraImages?:ValueTypes["ImageInput"][], + content?:string, + seo?:ValueTypes["SEOSimpleInput"], + urls?:string, + workspaceType:ValueTypes["WorkspaceTypeEnum"], + pricingPerHour:number, + currency:ValueTypes["AvailableCurrency"], + maxCapacity:number, + minStay:number, + maxStay?:number, + availability:ValueTypes["AvailabilityInput"], + mainImages:ValueTypes["MainImage"], + equipmentIds:string[], + featureIds:string[], + address:ValueTypes["AddressStrictInput"] +}; + ["SEOSimpleInput"]: { + title:string, + description:string, + keywords?:string, + thumbnail?:ValueTypes["ImageInput"] +}; + ["AvailabilityInput"]: { + dates:ValueTypes["DateRangeComponentInput"], + hours?:ValueTypes["HoursInput"][], + exceptions?:ValueTypes["DateExceptionComponentInput"][], + noWeekend?:boolean +}; + ["HoursInput"]: { + from:string, + to:string +}; + ["DateExceptionComponentInput"]: { + dates:ValueTypes["DateRangeComponentInput"], + hours?:ValueTypes["HoursInput"][], + allDay?:boolean +}; + ["MainImage"]: { + main?:ValueTypes["ImageInput"], + one?:ValueTypes["ImageInput"], + two?:ValueTypes["ImageInput"], + three?:ValueTypes["ImageInput"], + four?:ValueTypes["ImageInput"] +}; + ["EditWorkspaceInput"]: { + title?:string, + teaser?:string, + cover?:ValueTypes["ImageInput"], + thumbnail?:ValueTypes["ImageInput"], + extraImages?:ValueTypes["ImageInput"][], + content?:string, + seo?:ValueTypes["SEOSimpleInput"], + urls?:string, + workspaceType?:ValueTypes["WorkspaceTypeEnum"], + pricingPerHour?:number, + currency:ValueTypes["AvailableCurrency"], + maxCapacity?:number, + minStay?:number, + maxStay?:number, + availability?:ValueTypes["AvailabilityInput"], + mainImages:ValueTypes["MainImage"], + equipmentIds?:string[], + featureIds?:string[], + address?:ValueTypes["AddressStrictInput"] +}; + ["CheckoutInput"]: { + contactInfo:ValueTypes["BuyerInfo"], + billingInfo:ValueTypes["BuyerInfo"], + token?:string +}; + ["BuyerInfo"]: { + firstName:string, + lastName:string, + email:string, + company?:string, + vatNumber?:string, + address:ValueTypes["AddressInput"] +}; + ["NewRoomInput"]: { + accountIds:string[] +}; + ["AddPeopleToRoomInput"]: { + roomId:string, + accountIds:string[] +}; + ["RemovePeopleFromRoomInput"]: { + roomId:string, + accountIds:string[] +}; + ["BookingNewInputSchema"]: { + durationType:ValueTypes["DurationTypeEnum"], + startToEnd?:ValueTypes["DateRangeComponentInput"], + slot?:ValueTypes["SlotComponentInput"], + capacity?:number, + comments?:string, + ressourceModel:ValueTypes["BookingsRessourceEnum"], + ressourceId:string, + checkoutInfo?:ValueTypes["CheckoutEngineInputI"] +}; + ["SlotComponentInput"]: { + startDate:ValueTypes["DateTime"], + endDate:ValueTypes["DateTime"], + startTime:string, + endTime:string +}; + ["CheckoutEngineInputI"]: { + contactInfo:ValueTypes["BuyerInfo"], + billingInfo:ValueTypes["BuyerInfo"], + token?:string +}; + ["NewMessageInput"]: { + roomId:string, + message:string, + file?:ValueTypes["ImageInput"] +}; + ["EditMessageInput"]: { + message:string +} + } + +export type ModelTypes = { + ["Query"]: { + me:ModelTypes["AccountModel"], + login:ModelTypes["FirebaseTokenResult"], + magicLink:ModelTypes["SimpleResult"], + placesAutocomplete:ModelTypes["JSONObject"][], + placesGeocode:ModelTypes["JSONObject"], + placesGeocodeFromAddress:ModelTypes["JSONObject"], + categoriesGetOne:ModelTypes["CategoryModel"], + categoriesGetMany:ModelTypes["CategoryModel"][], + categoriesGetCount:number, + listsGetOne:ModelTypes["ListModel"], + listsGetMany:ModelTypes["ListModel"][], + listsGetCount:number, + accountsDashboardLinkStripe:string, + workspacesGetOne:ModelTypes["WorkspaceModel"], + workspacesGetMany:ModelTypes["WorkspaceModel"][], + workspacesGetCount:number, + workspacesSearchMany:ModelTypes["WorkspaceModel"][], + workspacesGetMine:ModelTypes["WorkspaceModel"][], + workspacesGetMineCount:number, + bookingsGetOne:ModelTypes["BookingEngineSchema"], + bookingsGetMany:ModelTypes["BookingEngineSchema"][], + bookingsGetCount:number, + myCurrentOrder:ModelTypes["OrderEngineSchema"], + myOrdersGetOne:ModelTypes["OrderEngineSchema"], + myOrdersGetMany:ModelTypes["OrderEngineSchema"][], + myOrdersGetManyCount:number, + roomsGetOne:ModelTypes["RoomModel"], + roomsGetMany:ModelTypes["RoomModel"][], + roomsGetMessages:ModelTypes["MessageModel"][] +}; + ["AccountModel"]: { + _id:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt:ModelTypes["DateTime"], + updatedAt:ModelTypes["DateTime"], + r:string[], + w:string[], + d:string[], + types:ModelTypes["AccountTypeEnum"][], + organisationIds?:string[], + email:string, + userName?:string, + firstName?:string, + lastName?:string, + profilePicture?:ModelTypes["ImageComponent"], + phoneNumber?:string, + gender?:ModelTypes["AccountGenderEnum"], + birthDate?:ModelTypes["DateTime"], + getAge?:number, + address?:ModelTypes["AddressComponent"], + language?:ModelTypes["AvailableTranslation"], + settings?:ModelTypes["SettingsComponent"][], + securityCheck:boolean, + paymentInfo?:ModelTypes["AccountPaymentInfo"], + payoutInfo?:ModelTypes["AccountPayoutInfo"], + terms:boolean, + favLikes?:(ModelTypes["FavoriteComponent"] | undefined)[], + getFavorites:ModelTypes["FavoriteComponent"][] +}; + /** The javascript `Date` as string. Type represents date and time as the ISO Date string. */ +["DateTime"]:any; + ["AccountTypeEnum"]: GraphQLTypes["AccountTypeEnum"]; + ["ImageComponent"]: { + title?:string, + fileType?:string, + large:string, + medium?:string, + small?:string +}; + ["AccountGenderEnum"]: GraphQLTypes["AccountGenderEnum"]; + ["AddressComponent"]: { + number?:string, + street?:string, + streetBis?:string, + floor?:string, + box?:string, + zip?:string, + state?:string, + city:string, + country:string +}; + ["AvailableTranslation"]: GraphQLTypes["AvailableTranslation"]; + ["SettingsComponent"]: { + type:ModelTypes["SettingsTypeEnum"], + email?:boolean, + pushNotifications?:boolean, + sms?:boolean +}; + ["SettingsTypeEnum"]: GraphQLTypes["SettingsTypeEnum"]; + ["AccountPaymentInfo"]: { + stripeInfo?:ModelTypes["StripeInfo"], + paymentMethods?:ModelTypes["PaymentMethodDetail"][], + billingInfos?:ModelTypes["BillingAddressComponent"][] +}; + ["StripeInfo"]: { + customerId:string +}; + ["PaymentMethodDetail"]: { + _id:string, + sPayMethodId:string, + default:boolean, + nameOnCard:string, + type:ModelTypes["cardTypeEnum"], + cardInfo?:ModelTypes["CardInfo"] +}; + ["cardTypeEnum"]: GraphQLTypes["cardTypeEnum"]; + ["CardInfo"]: { + brand:ModelTypes["cardBrandEnum"], + country:string, + exp_month:number, + exp_year:number, + last4:string +}; + ["cardBrandEnum"]: GraphQLTypes["cardBrandEnum"]; + ["BillingAddressComponent"]: { + firstName:string, + lastName:string, + address:ModelTypes["AddressStrictComponent"], + email?:string, + default?:boolean, + customTags?:ModelTypes["JSONObject"], + company?:string, + vatNumber?:string, + fiscalForm?:string, + _id:string +}; + ["AddressStrictComponent"]: { + number:string, + street:string, + zip:string, + city:string, + country:string, + streetBis?:string, + floor?:string, + box?:string, + state?:string +}; + /** The `JSONObject` scalar type represents JSON objects as specified by [ECMA-404](http://www.ecma-international.org/publications/files/ECMA-ST/ECMA-404.pdf). */ +["JSONObject"]:any; + ["AccountPayoutInfo"]: { + stripeInfo?:ModelTypes["StripePayoutInfo"] +}; + ["StripePayoutInfo"]: { + accountId:string, + chargesEnabled:boolean, + payoutsEnabled:boolean, + detailsSubmitted:boolean +}; + ["FavoriteComponent"]: { + ressourceId:string, + ressourceType:ModelTypes["RessourceEnum"], + addedAt:ModelTypes["DateTime"], + favorite?:boolean, + liked?:boolean +}; + ["RessourceEnum"]: GraphQLTypes["RessourceEnum"]; + ["FirebaseTokenResult"]: { + localId:string, + email?:string, + displayName?:string, + idToken:string, + registered?:boolean, + refreshToken:string, + expiresIn:string +}; + ["LoginInput"]: GraphQLTypes["LoginInput"]; + ["SimpleResult"]: { + message:string +}; + ["ResetPasswordInput"]: GraphQLTypes["ResetPasswordInput"]; + ["CountryCodesComponentEnum"]: GraphQLTypes["CountryCodesComponentEnum"]; + ["CategoryModel"]: { + _id:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt:ModelTypes["DateTime"], + updatedAt:ModelTypes["DateTime"], + r:string[], + w:string[], + d:string[], + title:ModelTypes["TranslatableComponent"], + teaser?:ModelTypes["TranslatableComponent"], + cover?:ModelTypes["ImageComponent"], + thumbnail?:ModelTypes["ImageComponent"], + extraImages?:ModelTypes["ImageComponent"][], + content?:ModelTypes["TranslatableComponent"], + seo?:ModelTypes["SEOField"], + urls?:ModelTypes["TranslatableComponent"], + ressourceType:ModelTypes["RessourceEnum"], + colorCode?:string +}; + ["TranslatableComponent"]: { + en?:string, + fr:string, + nl?:string, + de?:string +}; + ["SEOField"]: { + title:ModelTypes["TranslatableComponent"], + description:ModelTypes["TranslatableComponent"], + keywords?:ModelTypes["TranslatableComponent"], + thumbnail?:ModelTypes["ImageComponent"] +}; + ["GetArgs"]: GraphQLTypes["GetArgs"]; + ["ListModel"]: { + _id:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt:ModelTypes["DateTime"], + updatedAt:ModelTypes["DateTime"], + r:string[], + w:string[], + d:string[], + title:ModelTypes["TranslatableComponent"], + teaser?:ModelTypes["TranslatableComponent"], + cover?:ModelTypes["ImageComponent"], + thumbnail?:ModelTypes["ImageComponent"], + extraImages?:ModelTypes["ImageComponent"][], + content?:ModelTypes["TranslatableComponent"], + seo?:ModelTypes["SEOField"], + urls?:ModelTypes["TranslatableComponent"], + ressourceType:ModelTypes["ListEnum"], + colorCode?:string +}; + ["ListEnum"]: GraphQLTypes["ListEnum"]; + ["WorkspaceModel"]: { + _id:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt:ModelTypes["DateTime"], + updatedAt:ModelTypes["DateTime"], + r:string[], + w:string[], + d:string[], + title?:string, + teaser?:string, + cover?:ModelTypes["ImageComponent"], + thumbnail?:ModelTypes["ImageComponent"], + extraImages?:ModelTypes["ImageComponent"][], + content?:string, + seo?:ModelTypes["SEOSimpleField"], + urls?:string, + place:ModelTypes["PlaceComponent"], + dist?:ModelTypes["GeolocDistComponent"], + getDistFromLocation?:number, + workspaceType:ModelTypes["WorkspaceTypeEnum"], + pricingPerHour:number, + currency:ModelTypes["AvailableCurrency"], + maxCapacity:number, + minStay:number, + maxStay?:number, + availability:ModelTypes["AvailabilityComponent"], + mainImages:ModelTypes["MainImageComponent"], + equipmentIds:string[], + featureIds:string[], + address:ModelTypes["AddressComponent"], + finalPrice:number, + getEquipments?:(ModelTypes["ListModel"] | undefined)[], + getFeatures?:(ModelTypes["ListModel"] | undefined)[], + getBookings?:(ModelTypes["BookingSchema"] | undefined)[], + getOwner?:ModelTypes["WorkspaceOwnerComponent"] +}; + ["SEOSimpleField"]: { + title:string, + description:string, + keywords?:string, + thumbnail?:ModelTypes["ImageComponent"] +}; + ["PlaceComponent"]: { + placeId:string, + address?:ModelTypes["AddressComponent"], + loc:ModelTypes["LocComponent"], + formattedAddress:string +}; + ["LocComponent"]: { + type:string, + coordinates:number[] +}; + ["GeolocDistComponent"]: { + calculated?:number, + location?:ModelTypes["LocComponent"] +}; + ["WorkspaceTypeEnum"]: GraphQLTypes["WorkspaceTypeEnum"]; + ["AvailableCurrency"]: GraphQLTypes["AvailableCurrency"]; + ["AvailabilityComponent"]: { + dates:ModelTypes["DateRangeComponent"], + hours?:ModelTypes["HourComponent"][], + exceptions?:ModelTypes["DateExceptionComponent"][], + noWeekend?:boolean +}; + ["DateRangeComponent"]: { + startDate:ModelTypes["DateTime"], + endDate:ModelTypes["DateTime"] +}; + ["HourComponent"]: { + from:string, + to:string +}; + ["DateExceptionComponent"]: { + dates:ModelTypes["DateRangeComponent"], + hours?:ModelTypes["HourComponent"][], + allDay?:boolean +}; + ["MainImageComponent"]: { + main?:ModelTypes["ImageComponent"], + one?:ModelTypes["ImageComponent"], + two?:ModelTypes["ImageComponent"], + three?:ModelTypes["ImageComponent"], + four?:ModelTypes["ImageComponent"] +}; + ["BookingSchema"]: { + durationType:ModelTypes["DurationTypeEnum"], + startToEnd?:ModelTypes["DateRangeComponent"], + slot?:ModelTypes["SlotComponent"], + capacity:number, + comments?:string, + status:ModelTypes["StatusEnum"], + paths?:ModelTypes["EnginePathComponent"][], + ownerId:string, + dates:ModelTypes["DateRangeComponent"][], + _id:string, + organisationId?:string, + by?:ModelTypes["MetaBy"], + permissions?:ModelTypes["MetaPermissions"], + createdAt:ModelTypes["DateTime"], + updatedAt:ModelTypes["DateTime"], + tagsIds?:string[] +}; + ["IEngineSchema"]: ModelTypes["BookingSchema"] | ModelTypes["BookingEngineSchema"] | ModelTypes["OrderEngineSchema"]; + ["EnginePathComponent"]: { + ressourceModel:ModelTypes["ModelLoadersEnum"], + ressourceId:string +}; + ["ModelLoadersEnum"]: GraphQLTypes["ModelLoadersEnum"]; + ["MetaBy"]: { + createdBy?:string, + updatedBy?:string, + deletedBy?:string +}; + ["MetaPermissions"]: { + r:string[], + w:string[], + d:string[] +}; + ["DurationTypeEnum"]: GraphQLTypes["DurationTypeEnum"]; + ["SlotComponent"]: { + startDate:ModelTypes["DateTime"], + endDate:ModelTypes["DateTime"], + startTime:string, + endTime:string +}; + ["StatusEnum"]: GraphQLTypes["StatusEnum"]; + ["BookingsRessourceEnum"]: GraphQLTypes["BookingsRessourceEnum"]; + ["RangeType"]: GraphQLTypes["RangeType"]; + ["DateTabType"]: GraphQLTypes["DateTabType"]; + ["DateRangeComponentInput"]: GraphQLTypes["DateRangeComponentInput"]; + ["WorkspaceOwnerComponent"]: { + firstName?:string, + lastName?:string +}; + ["GeolocSearchInput"]: GraphQLTypes["GeolocSearchInput"]; + ["GeolocAddressSearchInput"]: GraphQLTypes["GeolocAddressSearchInput"]; + ["BookingEngineSchema"]: { + durationType:ModelTypes["DurationTypeEnum"], + startToEnd?:ModelTypes["DateRangeComponent"], + slot?:ModelTypes["SlotComponent"], + capacity?:number, + comments?:string, + status:ModelTypes["StatusEnum"], + paths?:ModelTypes["EnginePathComponent"][], + ownerId:string, + dates:ModelTypes["DateRangeComponent"][], + owner:ModelTypes["AccountModel"], + _id:string, + organisationId?:string, + by?:ModelTypes["MetaBy"], + permissions?:ModelTypes["MetaPermissions"], + createdAt:ModelTypes["DateTime"], + updatedAt:ModelTypes["DateTime"], + tagsIds?:string[], + workspace?:ModelTypes["WorkspaceModel"], + room?:ModelTypes["RoomModel"] +}; + ["RoomModel"]: { + _id:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt:ModelTypes["DateTime"], + updatedAt:ModelTypes["DateTime"], + r:string[], + w:string[], + d:string[], + title?:string, + stats:ModelTypes["RoomsStats"], + getAccounts?:ModelTypes["AccountModel"][], + getMessages?:ModelTypes["MessageModel"][], + lastMessage?:ModelTypes["DateTime"] +}; + ["RoomsStats"]: { + unreadCounts:ModelTypes["UnreadCount"][] +}; + ["UnreadCount"]: { + accountId:string, + count:number +}; + ["MessageModel"]: { + _id:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt:ModelTypes["DateTime"], + updatedAt:ModelTypes["DateTime"], + r:string[], + w:string[], + d:string[], + roomId:string, + notified?:boolean, + message:string, + edited:boolean, + deleted:boolean, + file?:ModelTypes["ImageComponent"], + sentBy:string, + readBy:ModelTypes["MessageReadBy"][] +}; + ["MessageReadBy"]: { + accountId:string, + readAt:ModelTypes["DateTime"] +}; + ["OrderEngineSchema"]: { + lines:ModelTypes["LineItemSchema"][], + localeInfo:ModelTypes["LocaleInfo"], + currency:ModelTypes["CurrencyEnum"], + vatExempt:boolean, + accountId:string, + contactInfo?:ModelTypes["BuyerInfoComponent"], + billingInfo?:ModelTypes["BuyerInfoComponent"], + promoId?:string, + invoiceInfo?:ModelTypes["InvoiceInfo"], + providerOrderItems?:ModelTypes["ProviderSchema"][], + orderName?:ModelTypes["TranslatableComponent"], + orderStatus:ModelTypes["OrderStatusEnum"], + paymentIntent:ModelTypes["PaymentIntentInfo"], + paymentStatus?:ModelTypes["PaymentStatusEnum"], + paymentInfo?:ModelTypes["PaymentInfo"], + subTotalPrice:number, + vatPrice:number, + finalPrice:number, + totalPrice:number, + promo?:ModelTypes["PromoModel"], + _id:string, + organisationId?:string, + paths?:ModelTypes["EnginePathComponent"][], + by?:ModelTypes["MetaBy"], + permissions?:ModelTypes["MetaPermissions"], + createdAt:ModelTypes["DateTime"], + updatedAt:ModelTypes["DateTime"], + tagsIds?:string[], + getBookingInput?:ModelTypes["BookingNewInput"] +}; + ["LineItemSchema"]: { + _id:string, + productRessource:ModelTypes["ProductRessourceEnum"], + productId:string, + title:string, + price:number, + salesPrice:number, + quantity:number, + vatClassId:string, + parentId?:string, + addOns?:ModelTypes["AddOnComponent"][], + localeInfo?:ModelTypes["LocaleInfo"], + promoId?:string, + finalPrice?:number, + getProduct?:ModelTypes["ProductUnion"], + getLinePrice:number, + getLineVatPrice:number +}; + ["ProductRessourceEnum"]: GraphQLTypes["ProductRessourceEnum"]; + ["AddOnComponent"]: { + addOnId:string, + quantity:number +}; + ["LocaleInfo"]: { + countryCode?:string, + locale?:ModelTypes["AvailableTranslation"] +}; + ["ProductUnion"]:ModelTypes["WorkspaceModel"]; + ["CurrencyEnum"]: GraphQLTypes["CurrencyEnum"]; + ["BuyerInfoComponent"]: { + firstName:string, + lastName:string, + email:string, + company?:string, + vatNumber?:string, + address:ModelTypes["AddressComponent"] +}; + ["InvoiceInfo"]: { + provider:ModelTypes["InvoicingProvider"], + invoiceId:string, + invoiceNumber:number +}; + ["InvoicingProvider"]: GraphQLTypes["InvoicingProvider"]; + ["ProviderSchema"]: { + _id:string, + organisationId:string, + orderStatus:ModelTypes["OrderBusinessStatusEnum"], + lines:ModelTypes["LineItemSchema"][] +}; + ["OrderBusinessStatusEnum"]: GraphQLTypes["OrderBusinessStatusEnum"]; + ["OrderStatusEnum"]: GraphQLTypes["OrderStatusEnum"]; + ["PaymentIntentInfo"]: { + provider:ModelTypes["PaymentProvider"], + stripePaymentIntentData:ModelTypes["StripePaymentIntentData"] +}; + ["PaymentProvider"]: GraphQLTypes["PaymentProvider"]; + ["StripePaymentIntentData"]: { + id:string, + client_secret?:string, + currency:string, + customer:string, + status:string +}; + ["PaymentStatusEnum"]: GraphQLTypes["PaymentStatusEnum"]; + ["PaymentInfo"]: { + provider:ModelTypes["PaymentProvider"], + transactionId:string +}; + ["PromoModel"]: { + _id:string, + organisationId?:string, + createdBy?:string, + updatedBy?:string, + deletedBy?:string, + createdAt:ModelTypes["DateTime"], + updatedAt:ModelTypes["DateTime"], + r:string[], + w:string[], + d:string[], + code:string, + description?:ModelTypes["TranslatableComponent"], + type:ModelTypes["PromoType"], + value:number, + validity?:ModelTypes["DateTime"], + cummulable:boolean, + usageLimit?:number, + usage:number +}; + ["PromoType"]: GraphQLTypes["PromoType"]; + ["BookingNewInput"]: { + durationType:ModelTypes["DurationTypeEnum"], + startToEnd?:ModelTypes["DateRangeComponent"], + slot?:ModelTypes["SlotComponent"], + capacity?:number, + comments?:string, + ressourceModel:ModelTypes["BookingsRessourceEnum"], + ressourceId:string, + checkoutInfo?:ModelTypes["CheckoutEngineInput"] +}; + ["CheckoutEngineInput"]: { + contactInfo:ModelTypes["BuyerInfoComponent"], + billingInfo:ModelTypes["BuyerInfoComponent"], + token?:string +}; + ["Mutation"]: { + updateMe:ModelTypes["AccountModel"], + updateMeEmail:ModelTypes["AccountModel"], + updateMePassword:ModelTypes["AccountModel"], + resetPassword:ModelTypes["SimpleResult"], + registerGuest:ModelTypes["AccountModel"], + register:ModelTypes["AccountModel"], + accountsBillingInfosAddOne:ModelTypes["BillingAddressComponent"], + accountsBillingInfosEditOne:ModelTypes["BillingAddressComponent"], + accountsBillingInfosDeleteOne:string, + accountsPaymentMethodsAddOne:ModelTypes["PaymentMethodDetail"], + accountsPaymentMethodsMarkasDefault:ModelTypes["PaymentMethodDetail"], + accountsPaymentMethodsDeleteOne:string, + accountsLinkWithStripe:string, + accountsRefreshInfoStripe:ModelTypes["AccountModel"], + accountsFavoritesAddOne:ModelTypes["FavoriteComponent"], + accountsFavoritesRemoveOne:ModelTypes["SuccessResponse"], + workspacesAddOne:ModelTypes["WorkspaceModel"], + workspacesEditOne:ModelTypes["WorkspaceModel"], + workspacesDeleteOne:ModelTypes["WorkspaceModel"], + finaliseCheckoutWithStripe:ModelTypes["OrderEngineSchema"], + removeCurrentOrder:ModelTypes["OrderEngineSchema"], + validateOrder:ModelTypes["OrderEngineSchema"], + roomsCreateOne:ModelTypes["RoomModel"], + roomsAddPeople:ModelTypes["RoomModel"], + roomsRemovePeople:ModelTypes["RoomModel"], + roomsMarkAllMessageAsRead:ModelTypes["RoomModel"], + reserveOneBooking:ModelTypes["OrderEngineSchema"], + roomsAddOneMessage:ModelTypes["MessageModel"], + roomsEditOneMessage:ModelTypes["MessageModel"], + roomsMarkMessageAsRead:ModelTypes["MessageModel"], + roomsDeleteOneMessage:ModelTypes["MessageModel"] +}; + ["EditAccountInput"]: GraphQLTypes["EditAccountInput"]; + ["ImageInput"]: GraphQLTypes["ImageInput"]; + ["AddressInput"]: GraphQLTypes["AddressInput"]; + ["SettingsInput"]: GraphQLTypes["SettingsInput"]; + ["NewEmailInput"]: GraphQLTypes["NewEmailInput"]; + ["NewPasswordInput"]: GraphQLTypes["NewPasswordInput"]; + ["LinkEmailInput"]: GraphQLTypes["LinkEmailInput"]; + ["NewAccountInput"]: GraphQLTypes["NewAccountInput"]; + ["BillingAddressInput"]: GraphQLTypes["BillingAddressInput"]; + ["AddressStrictInput"]: GraphQLTypes["AddressStrictInput"]; + ["PaymentMethodInput"]: GraphQLTypes["PaymentMethodInput"]; + ["AddInput"]: GraphQLTypes["AddInput"]; + ["SuccessResponse"]: { + success:boolean +}; + ["RemoveInput"]: GraphQLTypes["RemoveInput"]; + ["NewWorkspaceInput"]: GraphQLTypes["NewWorkspaceInput"]; + ["SEOSimpleInput"]: GraphQLTypes["SEOSimpleInput"]; + ["AvailabilityInput"]: GraphQLTypes["AvailabilityInput"]; + ["HoursInput"]: GraphQLTypes["HoursInput"]; + ["DateExceptionComponentInput"]: GraphQLTypes["DateExceptionComponentInput"]; + ["MainImage"]: GraphQLTypes["MainImage"]; + ["EditWorkspaceInput"]: GraphQLTypes["EditWorkspaceInput"]; + ["CheckoutInput"]: GraphQLTypes["CheckoutInput"]; + ["BuyerInfo"]: GraphQLTypes["BuyerInfo"]; + ["NewRoomInput"]: GraphQLTypes["NewRoomInput"]; + ["AddPeopleToRoomInput"]: GraphQLTypes["AddPeopleToRoomInput"]; + ["RemovePeopleFromRoomInput"]: GraphQLTypes["RemovePeopleFromRoomInput"]; + ["BookingNewInputSchema"]: GraphQLTypes["BookingNewInputSchema"]; + ["SlotComponentInput"]: GraphQLTypes["SlotComponentInput"]; + ["CheckoutEngineInputI"]: GraphQLTypes["CheckoutEngineInputI"]; + ["NewMessageInput"]: GraphQLTypes["NewMessageInput"]; + ["EditMessageInput"]: GraphQLTypes["EditMessageInput"] + } + +export type GraphQLTypes = { + ["Query"]: { + __typename: "Query", + me: GraphQLTypes["AccountModel"], + login: GraphQLTypes["FirebaseTokenResult"], + magicLink: GraphQLTypes["SimpleResult"], + placesAutocomplete: Array, + placesGeocode: GraphQLTypes["JSONObject"], + placesGeocodeFromAddress: GraphQLTypes["JSONObject"], + categoriesGetOne: GraphQLTypes["CategoryModel"], + categoriesGetMany: Array, + categoriesGetCount: number, + listsGetOne: GraphQLTypes["ListModel"], + listsGetMany: Array, + listsGetCount: number, + accountsDashboardLinkStripe: string, + workspacesGetOne: GraphQLTypes["WorkspaceModel"], + workspacesGetMany: Array, + workspacesGetCount: number, + workspacesSearchMany: Array, + workspacesGetMine: Array, + workspacesGetMineCount: number, + bookingsGetOne: GraphQLTypes["BookingEngineSchema"], + bookingsGetMany: Array, + bookingsGetCount: number, + myCurrentOrder: GraphQLTypes["OrderEngineSchema"], + myOrdersGetOne: GraphQLTypes["OrderEngineSchema"], + myOrdersGetMany: Array, + myOrdersGetManyCount: number, + roomsGetOne: GraphQLTypes["RoomModel"], + roomsGetMany: Array, + roomsGetMessages: Array +}; + ["AccountModel"]: { + __typename: "AccountModel", + _id: string, + organisationId: string | null, + createdBy: string | null, + updatedBy: string | null, + deletedBy: string | null, + createdAt: GraphQLTypes["DateTime"], + updatedAt: GraphQLTypes["DateTime"], + r: Array, + w: Array, + d: Array, + types: Array, + organisationIds: Array | null, + email: string, + userName: string | null, + firstName: string | null, + lastName: string | null, + profilePicture: GraphQLTypes["ImageComponent"] | null, + phoneNumber: string | null, + gender: GraphQLTypes["AccountGenderEnum"] | null, + birthDate: GraphQLTypes["DateTime"] | null, + getAge: number | null, + address: GraphQLTypes["AddressComponent"] | null, + language: GraphQLTypes["AvailableTranslation"] | null, + settings: Array | null, + securityCheck: boolean, + paymentInfo: GraphQLTypes["AccountPaymentInfo"] | null, + payoutInfo: GraphQLTypes["AccountPayoutInfo"] | null, + terms: boolean, + favLikes: Array | null, + getFavorites: Array +}; + /** The javascript `Date` as string. Type represents date and time as the ISO Date string. */ +["DateTime"]:any; + ["AccountTypeEnum"]: AccountTypeEnum; + ["ImageComponent"]: { + __typename: "ImageComponent", + title: string | null, + fileType: string | null, + large: string, + medium: string | null, + small: string | null +}; + ["AccountGenderEnum"]: AccountGenderEnum; + ["AddressComponent"]: { + __typename: "AddressComponent", + number: string | null, + street: string | null, + streetBis: string | null, + floor: string | null, + box: string | null, + zip: string | null, + state: string | null, + city: string, + country: string +}; + ["AvailableTranslation"]: AvailableTranslation; + ["SettingsComponent"]: { + __typename: "SettingsComponent", + type: GraphQLTypes["SettingsTypeEnum"], + email: boolean | null, + pushNotifications: boolean | null, + sms: boolean | null +}; + ["SettingsTypeEnum"]: SettingsTypeEnum; + ["AccountPaymentInfo"]: { + __typename: "AccountPaymentInfo", + stripeInfo: GraphQLTypes["StripeInfo"] | null, + paymentMethods: Array | null, + billingInfos: Array | null +}; + ["StripeInfo"]: { + __typename: "StripeInfo", + customerId: string +}; + ["PaymentMethodDetail"]: { + __typename: "PaymentMethodDetail", + _id: string, + sPayMethodId: string, + default: boolean, + nameOnCard: string, + type: GraphQLTypes["cardTypeEnum"], + cardInfo: GraphQLTypes["CardInfo"] | null +}; + ["cardTypeEnum"]: cardTypeEnum; + ["CardInfo"]: { + __typename: "CardInfo", + brand: GraphQLTypes["cardBrandEnum"], + country: string, + exp_month: number, + exp_year: number, + last4: string +}; + ["cardBrandEnum"]: cardBrandEnum; + ["BillingAddressComponent"]: { + __typename: "BillingAddressComponent", + firstName: string, + lastName: string, + address: GraphQLTypes["AddressStrictComponent"], + email: string | null, + default: boolean | null, + customTags: GraphQLTypes["JSONObject"] | null, + company: string | null, + vatNumber: string | null, + fiscalForm: string | null, + _id: string +}; + ["AddressStrictComponent"]: { + __typename: "AddressStrictComponent", + number: string, + street: string, + zip: string, + city: string, + country: string, + streetBis: string | null, + floor: string | null, + box: string | null, + state: string | null +}; + /** The `JSONObject` scalar type represents JSON objects as specified by [ECMA-404](http://www.ecma-international.org/publications/files/ECMA-ST/ECMA-404.pdf). */ +["JSONObject"]:any; + ["AccountPayoutInfo"]: { + __typename: "AccountPayoutInfo", + stripeInfo: GraphQLTypes["StripePayoutInfo"] | null +}; + ["StripePayoutInfo"]: { + __typename: "StripePayoutInfo", + accountId: string, + chargesEnabled: boolean, + payoutsEnabled: boolean, + detailsSubmitted: boolean +}; + ["FavoriteComponent"]: { + __typename: "FavoriteComponent", + ressourceId: string, + ressourceType: GraphQLTypes["RessourceEnum"], + addedAt: GraphQLTypes["DateTime"], + favorite: boolean | null, + liked: boolean | null +}; + ["RessourceEnum"]: RessourceEnum; + ["FirebaseTokenResult"]: { + __typename: "FirebaseTokenResult", + localId: string, + email: string | null, + displayName: string | null, + idToken: string, + registered: boolean | null, + refreshToken: string, + expiresIn: string +}; + ["LoginInput"]: { + email: string, + password: string +}; + ["SimpleResult"]: { + __typename: "SimpleResult", + message: string +}; + ["ResetPasswordInput"]: { + email: string +}; + ["CountryCodesComponentEnum"]: CountryCodesComponentEnum; + ["CategoryModel"]: { + __typename: "CategoryModel", + _id: string, + organisationId: string | null, + createdBy: string | null, + updatedBy: string | null, + deletedBy: string | null, + createdAt: GraphQLTypes["DateTime"], + updatedAt: GraphQLTypes["DateTime"], + r: Array, + w: Array, + d: Array, + title: GraphQLTypes["TranslatableComponent"], + teaser: GraphQLTypes["TranslatableComponent"] | null, + cover: GraphQLTypes["ImageComponent"] | null, + thumbnail: GraphQLTypes["ImageComponent"] | null, + extraImages: Array | null, + content: GraphQLTypes["TranslatableComponent"] | null, + seo: GraphQLTypes["SEOField"] | null, + urls: GraphQLTypes["TranslatableComponent"] | null, + ressourceType: GraphQLTypes["RessourceEnum"], + colorCode: string | null +}; + ["TranslatableComponent"]: { + __typename: "TranslatableComponent", + en: string | null, + fr: string, + nl: string | null, + de: string | null +}; + ["SEOField"]: { + __typename: "SEOField", + title: GraphQLTypes["TranslatableComponent"], + description: GraphQLTypes["TranslatableComponent"], + keywords: GraphQLTypes["TranslatableComponent"] | null, + thumbnail: GraphQLTypes["ImageComponent"] | null +}; + ["GetArgs"]: { + limit: number, + skip: number, + sort: string | null +}; + ["ListModel"]: { + __typename: "ListModel", + _id: string, + organisationId: string | null, + createdBy: string | null, + updatedBy: string | null, + deletedBy: string | null, + createdAt: GraphQLTypes["DateTime"], + updatedAt: GraphQLTypes["DateTime"], + r: Array, + w: Array, + d: Array, + title: GraphQLTypes["TranslatableComponent"], + teaser: GraphQLTypes["TranslatableComponent"] | null, + cover: GraphQLTypes["ImageComponent"] | null, + thumbnail: GraphQLTypes["ImageComponent"] | null, + extraImages: Array | null, + content: GraphQLTypes["TranslatableComponent"] | null, + seo: GraphQLTypes["SEOField"] | null, + urls: GraphQLTypes["TranslatableComponent"] | null, + ressourceType: GraphQLTypes["ListEnum"], + colorCode: string | null +}; + ["ListEnum"]: ListEnum; + ["WorkspaceModel"]: { + __typename: "WorkspaceModel", + _id: string, + organisationId: string | null, + createdBy: string | null, + updatedBy: string | null, + deletedBy: string | null, + createdAt: GraphQLTypes["DateTime"], + updatedAt: GraphQLTypes["DateTime"], + r: Array, + w: Array, + d: Array, + title: string | null, + teaser: string | null, + cover: GraphQLTypes["ImageComponent"] | null, + thumbnail: GraphQLTypes["ImageComponent"] | null, + extraImages: Array | null, + content: string | null, + seo: GraphQLTypes["SEOSimpleField"] | null, + urls: string | null, + place: GraphQLTypes["PlaceComponent"], + dist: GraphQLTypes["GeolocDistComponent"] | null, + getDistFromLocation: number | null, + workspaceType: GraphQLTypes["WorkspaceTypeEnum"], + pricingPerHour: number, + currency: GraphQLTypes["AvailableCurrency"], + maxCapacity: number, + minStay: number, + maxStay: number | null, + availability: GraphQLTypes["AvailabilityComponent"], + mainImages: GraphQLTypes["MainImageComponent"], + equipmentIds: Array, + featureIds: Array, + address: GraphQLTypes["AddressComponent"], + finalPrice: number, + getEquipments: Array | null, + getFeatures: Array | null, + getBookings: Array | null, + getOwner: GraphQLTypes["WorkspaceOwnerComponent"] | null +}; + ["SEOSimpleField"]: { + __typename: "SEOSimpleField", + title: string, + description: string, + keywords: string | null, + thumbnail: GraphQLTypes["ImageComponent"] | null +}; + ["PlaceComponent"]: { + __typename: "PlaceComponent", + placeId: string, + address: GraphQLTypes["AddressComponent"] | null, + loc: GraphQLTypes["LocComponent"], + formattedAddress: string +}; + ["LocComponent"]: { + __typename: "LocComponent", + type: string, + coordinates: Array +}; + ["GeolocDistComponent"]: { + __typename: "GeolocDistComponent", + calculated: number | null, + location: GraphQLTypes["LocComponent"] | null +}; + ["WorkspaceTypeEnum"]: WorkspaceTypeEnum; + ["AvailableCurrency"]: AvailableCurrency; + ["AvailabilityComponent"]: { + __typename: "AvailabilityComponent", + dates: GraphQLTypes["DateRangeComponent"], + hours: Array | null, + exceptions: Array | null, + noWeekend: boolean | null +}; + ["DateRangeComponent"]: { + __typename: "DateRangeComponent", + startDate: GraphQLTypes["DateTime"], + endDate: GraphQLTypes["DateTime"] +}; + ["HourComponent"]: { + __typename: "HourComponent", + from: string, + to: string +}; + ["DateExceptionComponent"]: { + __typename: "DateExceptionComponent", + dates: GraphQLTypes["DateRangeComponent"], + hours: Array | null, + allDay: boolean | null +}; + ["MainImageComponent"]: { + __typename: "MainImageComponent", + main: GraphQLTypes["ImageComponent"] | null, + one: GraphQLTypes["ImageComponent"] | null, + two: GraphQLTypes["ImageComponent"] | null, + three: GraphQLTypes["ImageComponent"] | null, + four: GraphQLTypes["ImageComponent"] | null +}; + ["BookingSchema"]: { + __typename: "BookingSchema", + durationType: GraphQLTypes["DurationTypeEnum"], + startToEnd: GraphQLTypes["DateRangeComponent"] | null, + slot: GraphQLTypes["SlotComponent"] | null, + capacity: number, + comments: string | null, + status: GraphQLTypes["StatusEnum"], + paths: Array | null, + ownerId: string, + dates: Array, + _id: string, + organisationId: string | null, + by: GraphQLTypes["MetaBy"] | null, + permissions: GraphQLTypes["MetaPermissions"] | null, + createdAt: GraphQLTypes["DateTime"], + updatedAt: GraphQLTypes["DateTime"], + tagsIds: Array | null +}; + ["IEngineSchema"]: { + __typename:"BookingSchema" | "BookingEngineSchema" | "OrderEngineSchema" + _id: string, + organisationId: string | null, + paths: Array | null, + by: GraphQLTypes["MetaBy"] | null, + permissions: GraphQLTypes["MetaPermissions"] | null, + createdAt: GraphQLTypes["DateTime"], + updatedAt: GraphQLTypes["DateTime"], + tagsIds: Array | null + ['...on BookingSchema']: '__union' & GraphQLTypes["BookingSchema"]; + ['...on BookingEngineSchema']: '__union' & GraphQLTypes["BookingEngineSchema"]; + ['...on OrderEngineSchema']: '__union' & GraphQLTypes["OrderEngineSchema"]; +}; + ["EnginePathComponent"]: { + __typename: "EnginePathComponent", + ressourceModel: GraphQLTypes["ModelLoadersEnum"], + ressourceId: string +}; + ["ModelLoadersEnum"]: ModelLoadersEnum; + ["MetaBy"]: { + __typename: "MetaBy", + createdBy: string | null, + updatedBy: string | null, + deletedBy: string | null +}; + ["MetaPermissions"]: { + __typename: "MetaPermissions", + r: Array, + w: Array, + d: Array +}; + ["DurationTypeEnum"]: DurationTypeEnum; + ["SlotComponent"]: { + __typename: "SlotComponent", + startDate: GraphQLTypes["DateTime"], + endDate: GraphQLTypes["DateTime"], + startTime: string, + endTime: string +}; + ["StatusEnum"]: StatusEnum; + ["BookingsRessourceEnum"]: BookingsRessourceEnum; + ["RangeType"]: RangeType; + ["DateTabType"]: DateTabType; + ["DateRangeComponentInput"]: { + startDate: GraphQLTypes["DateTime"], + endDate: GraphQLTypes["DateTime"] +}; + ["WorkspaceOwnerComponent"]: { + __typename: "WorkspaceOwnerComponent", + firstName: string | null, + lastName: string | null +}; + ["GeolocSearchInput"]: { + longitude: number, + latitude: number, + radius: number | null +}; + ["GeolocAddressSearchInput"]: { + formattedAddress: string, + radius: number | null +}; + ["BookingEngineSchema"]: { + __typename: "BookingEngineSchema", + durationType: GraphQLTypes["DurationTypeEnum"], + startToEnd: GraphQLTypes["DateRangeComponent"] | null, + slot: GraphQLTypes["SlotComponent"] | null, + capacity: number | null, + comments: string | null, + status: GraphQLTypes["StatusEnum"], + paths: Array | null, + ownerId: string, + dates: Array, + owner: GraphQLTypes["AccountModel"], + _id: string, + organisationId: string | null, + by: GraphQLTypes["MetaBy"] | null, + permissions: GraphQLTypes["MetaPermissions"] | null, + createdAt: GraphQLTypes["DateTime"], + updatedAt: GraphQLTypes["DateTime"], + tagsIds: Array | null, + workspace: GraphQLTypes["WorkspaceModel"] | null, + room: GraphQLTypes["RoomModel"] | null +}; + ["RoomModel"]: { + __typename: "RoomModel", + _id: string, + organisationId: string | null, + createdBy: string | null, + updatedBy: string | null, + deletedBy: string | null, + createdAt: GraphQLTypes["DateTime"], + updatedAt: GraphQLTypes["DateTime"], + r: Array, + w: Array, + d: Array, + title: string | null, + stats: GraphQLTypes["RoomsStats"], + getAccounts: Array | null, + getMessages: Array | null, + lastMessage: GraphQLTypes["DateTime"] | null +}; + ["RoomsStats"]: { + __typename: "RoomsStats", + unreadCounts: Array +}; + ["UnreadCount"]: { + __typename: "UnreadCount", + accountId: string, + count: number +}; + ["MessageModel"]: { + __typename: "MessageModel", + _id: string, + organisationId: string | null, + createdBy: string | null, + updatedBy: string | null, + deletedBy: string | null, + createdAt: GraphQLTypes["DateTime"], + updatedAt: GraphQLTypes["DateTime"], + r: Array, + w: Array, + d: Array, + roomId: string, + notified: boolean | null, + message: string, + edited: boolean, + deleted: boolean, + file: GraphQLTypes["ImageComponent"] | null, + sentBy: string, + readBy: Array +}; + ["MessageReadBy"]: { + __typename: "MessageReadBy", + accountId: string, + readAt: GraphQLTypes["DateTime"] +}; + ["OrderEngineSchema"]: { + __typename: "OrderEngineSchema", + lines: Array, + localeInfo: GraphQLTypes["LocaleInfo"], + currency: GraphQLTypes["CurrencyEnum"], + vatExempt: boolean, + accountId: string, + contactInfo: GraphQLTypes["BuyerInfoComponent"] | null, + billingInfo: GraphQLTypes["BuyerInfoComponent"] | null, + promoId: string | null, + invoiceInfo: GraphQLTypes["InvoiceInfo"] | null, + providerOrderItems: Array | null, + orderName: GraphQLTypes["TranslatableComponent"] | null, + orderStatus: GraphQLTypes["OrderStatusEnum"], + paymentIntent: GraphQLTypes["PaymentIntentInfo"], + paymentStatus: GraphQLTypes["PaymentStatusEnum"] | null, + paymentInfo: GraphQLTypes["PaymentInfo"] | null, + subTotalPrice: number, + vatPrice: number, + finalPrice: number, + totalPrice: number, + promo: GraphQLTypes["PromoModel"] | null, + _id: string, + organisationId: string | null, + paths: Array | null, + by: GraphQLTypes["MetaBy"] | null, + permissions: GraphQLTypes["MetaPermissions"] | null, + createdAt: GraphQLTypes["DateTime"], + updatedAt: GraphQLTypes["DateTime"], + tagsIds: Array | null, + getBookingInput: GraphQLTypes["BookingNewInput"] | null +}; + ["LineItemSchema"]: { + __typename: "LineItemSchema", + _id: string, + productRessource: GraphQLTypes["ProductRessourceEnum"], + productId: string, + title: string, + price: number, + salesPrice: number, + quantity: number, + vatClassId: string, + parentId: string | null, + addOns: Array | null, + localeInfo: GraphQLTypes["LocaleInfo"] | null, + promoId: string | null, + finalPrice: number | null, + getProduct: GraphQLTypes["ProductUnion"] | null, + getLinePrice: number, + getLineVatPrice: number +}; + ["ProductRessourceEnum"]: ProductRessourceEnum; + ["AddOnComponent"]: { + __typename: "AddOnComponent", + addOnId: string, + quantity: number +}; + ["LocaleInfo"]: { + __typename: "LocaleInfo", + countryCode: string | null, + locale: GraphQLTypes["AvailableTranslation"] | null +}; + ["ProductUnion"]:{ + ['...on WorkspaceModel']: '__union' & GraphQLTypes["WorkspaceModel"]; +}; + ["CurrencyEnum"]: CurrencyEnum; + ["BuyerInfoComponent"]: { + __typename: "BuyerInfoComponent", + firstName: string, + lastName: string, + email: string, + company: string | null, + vatNumber: string | null, + address: GraphQLTypes["AddressComponent"] +}; + ["InvoiceInfo"]: { + __typename: "InvoiceInfo", + provider: GraphQLTypes["InvoicingProvider"], + invoiceId: string, + invoiceNumber: number +}; + ["InvoicingProvider"]: InvoicingProvider; + ["ProviderSchema"]: { + __typename: "ProviderSchema", + _id: string, + organisationId: string, + orderStatus: GraphQLTypes["OrderBusinessStatusEnum"], + lines: Array +}; + ["OrderBusinessStatusEnum"]: OrderBusinessStatusEnum; + ["OrderStatusEnum"]: OrderStatusEnum; + ["PaymentIntentInfo"]: { + __typename: "PaymentIntentInfo", + provider: GraphQLTypes["PaymentProvider"], + stripePaymentIntentData: GraphQLTypes["StripePaymentIntentData"] +}; + ["PaymentProvider"]: PaymentProvider; + ["StripePaymentIntentData"]: { + __typename: "StripePaymentIntentData", + id: string, + client_secret: string | null, + currency: string, + customer: string, + status: string +}; + ["PaymentStatusEnum"]: PaymentStatusEnum; + ["PaymentInfo"]: { + __typename: "PaymentInfo", + provider: GraphQLTypes["PaymentProvider"], + transactionId: string +}; + ["PromoModel"]: { + __typename: "PromoModel", + _id: string, + organisationId: string | null, + createdBy: string | null, + updatedBy: string | null, + deletedBy: string | null, + createdAt: GraphQLTypes["DateTime"], + updatedAt: GraphQLTypes["DateTime"], + r: Array, + w: Array, + d: Array, + code: string, + description: GraphQLTypes["TranslatableComponent"] | null, + type: GraphQLTypes["PromoType"], + value: number, + validity: GraphQLTypes["DateTime"] | null, + cummulable: boolean, + usageLimit: number | null, + usage: number +}; + ["PromoType"]: PromoType; + ["BookingNewInput"]: { + __typename: "BookingNewInput", + durationType: GraphQLTypes["DurationTypeEnum"], + startToEnd: GraphQLTypes["DateRangeComponent"] | null, + slot: GraphQLTypes["SlotComponent"] | null, + capacity: number | null, + comments: string | null, + ressourceModel: GraphQLTypes["BookingsRessourceEnum"], + ressourceId: string, + checkoutInfo: GraphQLTypes["CheckoutEngineInput"] | null +}; + ["CheckoutEngineInput"]: { + __typename: "CheckoutEngineInput", + contactInfo: GraphQLTypes["BuyerInfoComponent"], + billingInfo: GraphQLTypes["BuyerInfoComponent"], + token: string | null +}; + ["Mutation"]: { + __typename: "Mutation", + updateMe: GraphQLTypes["AccountModel"], + updateMeEmail: GraphQLTypes["AccountModel"], + updateMePassword: GraphQLTypes["AccountModel"], + resetPassword: GraphQLTypes["SimpleResult"], + registerGuest: GraphQLTypes["AccountModel"], + register: GraphQLTypes["AccountModel"], + accountsBillingInfosAddOne: GraphQLTypes["BillingAddressComponent"], + accountsBillingInfosEditOne: GraphQLTypes["BillingAddressComponent"], + accountsBillingInfosDeleteOne: string, + accountsPaymentMethodsAddOne: GraphQLTypes["PaymentMethodDetail"], + accountsPaymentMethodsMarkasDefault: GraphQLTypes["PaymentMethodDetail"], + accountsPaymentMethodsDeleteOne: string, + accountsLinkWithStripe: string, + accountsRefreshInfoStripe: GraphQLTypes["AccountModel"], + accountsFavoritesAddOne: GraphQLTypes["FavoriteComponent"], + accountsFavoritesRemoveOne: GraphQLTypes["SuccessResponse"], + workspacesAddOne: GraphQLTypes["WorkspaceModel"], + workspacesEditOne: GraphQLTypes["WorkspaceModel"], + workspacesDeleteOne: GraphQLTypes["WorkspaceModel"], + finaliseCheckoutWithStripe: GraphQLTypes["OrderEngineSchema"], + removeCurrentOrder: GraphQLTypes["OrderEngineSchema"], + validateOrder: GraphQLTypes["OrderEngineSchema"], + roomsCreateOne: GraphQLTypes["RoomModel"], + roomsAddPeople: GraphQLTypes["RoomModel"], + roomsRemovePeople: GraphQLTypes["RoomModel"], + roomsMarkAllMessageAsRead: GraphQLTypes["RoomModel"], + reserveOneBooking: GraphQLTypes["OrderEngineSchema"], + roomsAddOneMessage: GraphQLTypes["MessageModel"], + roomsEditOneMessage: GraphQLTypes["MessageModel"], + roomsMarkMessageAsRead: GraphQLTypes["MessageModel"], + roomsDeleteOneMessage: GraphQLTypes["MessageModel"] +}; + ["EditAccountInput"]: { + userName: string | null, + firstName: string | null, + lastName: string | null, + profilePicture: GraphQLTypes["ImageInput"] | null, + phoneNumber: string | null, + gender: GraphQLTypes["AccountGenderEnum"] | null, + birthDate: GraphQLTypes["DateTime"] | null, + address: GraphQLTypes["AddressInput"] | null, + language: GraphQLTypes["AvailableTranslation"] | null, + settings: Array | null +}; + ["ImageInput"]: { + title: string | null, + fileType: string | null, + large: string, + medium: string | null, + small: string | null +}; + ["AddressInput"]: { + number: string | null, + street: string | null, + streetBis: string | null, + floor: string | null, + box: string | null, + zip: string | null, + state: string | null, + city: string, + country: string +}; + ["SettingsInput"]: { + type: GraphQLTypes["SettingsTypeEnum"], + email: boolean | null, + pushNotifications: boolean | null, + sms: boolean | null +}; + ["NewEmailInput"]: { + newEmail: string +}; + ["NewPasswordInput"]: { + oldPassword: string, + newPassword: string, + newPasswordConfirmation: string +}; + ["LinkEmailInput"]: { + email: string, + password: string, + idToken: string +}; + ["NewAccountInput"]: { + email: string, + password: string, + passwordConfirmation: string, + userName: string | null, + firstName: string | null, + lastName: string | null, + profilePicture: GraphQLTypes["ImageInput"] | null, + phoneNumber: string | null, + gender: GraphQLTypes["AccountGenderEnum"] | null, + birthDate: GraphQLTypes["DateTime"] | null, + address: GraphQLTypes["AddressInput"] | null, + language: GraphQLTypes["AvailableTranslation"] | null, + settings: Array | null, + terms: boolean +}; + ["BillingAddressInput"]: { + firstName: string, + lastName: string, + address: GraphQLTypes["AddressStrictInput"], + email: string | null, + default: boolean | null, + customTags: GraphQLTypes["JSONObject"] | null, + company: string | null, + vatNumber: string | null, + fiscalForm: string | null +}; + ["AddressStrictInput"]: { + number: string, + street: string, + zip: string, + city: string, + country: string, + streetBis: string | null, + floor: string | null, + box: string | null, + state: string | null +}; + ["PaymentMethodInput"]: { + sPayMethodId: string, + nameOnCard: string, + default: boolean | null +}; + ["AddInput"]: { + ressourceId: string, + ressourceType: GraphQLTypes["RessourceEnum"] +}; + ["SuccessResponse"]: { + __typename: "SuccessResponse", + success: boolean +}; + ["RemoveInput"]: { + ressourceId: string +}; + ["NewWorkspaceInput"]: { + title: string | null, + teaser: string | null, + cover: GraphQLTypes["ImageInput"] | null, + thumbnail: GraphQLTypes["ImageInput"] | null, + extraImages: Array | null, + content: string | null, + seo: GraphQLTypes["SEOSimpleInput"] | null, + urls: string | null, + workspaceType: GraphQLTypes["WorkspaceTypeEnum"], + pricingPerHour: number, + currency: GraphQLTypes["AvailableCurrency"], + maxCapacity: number, + minStay: number, + maxStay: number | null, + availability: GraphQLTypes["AvailabilityInput"], + mainImages: GraphQLTypes["MainImage"], + equipmentIds: Array, + featureIds: Array, + address: GraphQLTypes["AddressStrictInput"] +}; + ["SEOSimpleInput"]: { + title: string, + description: string, + keywords: string | null, + thumbnail: GraphQLTypes["ImageInput"] | null +}; + ["AvailabilityInput"]: { + dates: GraphQLTypes["DateRangeComponentInput"], + hours: Array | null, + exceptions: Array | null, + noWeekend: boolean | null +}; + ["HoursInput"]: { + from: string, + to: string +}; + ["DateExceptionComponentInput"]: { + dates: GraphQLTypes["DateRangeComponentInput"], + hours: Array | null, + allDay: boolean | null +}; + ["MainImage"]: { + main: GraphQLTypes["ImageInput"] | null, + one: GraphQLTypes["ImageInput"] | null, + two: GraphQLTypes["ImageInput"] | null, + three: GraphQLTypes["ImageInput"] | null, + four: GraphQLTypes["ImageInput"] | null +}; + ["EditWorkspaceInput"]: { + title: string | null, + teaser: string | null, + cover: GraphQLTypes["ImageInput"] | null, + thumbnail: GraphQLTypes["ImageInput"] | null, + extraImages: Array | null, + content: string | null, + seo: GraphQLTypes["SEOSimpleInput"] | null, + urls: string | null, + workspaceType: GraphQLTypes["WorkspaceTypeEnum"] | null, + pricingPerHour: number | null, + currency: GraphQLTypes["AvailableCurrency"], + maxCapacity: number | null, + minStay: number | null, + maxStay: number | null, + availability: GraphQLTypes["AvailabilityInput"] | null, + mainImages: GraphQLTypes["MainImage"], + equipmentIds: Array | null, + featureIds: Array | null, + address: GraphQLTypes["AddressStrictInput"] | null +}; + ["CheckoutInput"]: { + contactInfo: GraphQLTypes["BuyerInfo"], + billingInfo: GraphQLTypes["BuyerInfo"], + token: string | null +}; + ["BuyerInfo"]: { + firstName: string, + lastName: string, + email: string, + company: string | null, + vatNumber: string | null, + address: GraphQLTypes["AddressInput"] +}; + ["NewRoomInput"]: { + accountIds: Array +}; + ["AddPeopleToRoomInput"]: { + roomId: string, + accountIds: Array +}; + ["RemovePeopleFromRoomInput"]: { + roomId: string, + accountIds: Array +}; + ["BookingNewInputSchema"]: { + durationType: GraphQLTypes["DurationTypeEnum"], + startToEnd: GraphQLTypes["DateRangeComponentInput"] | null, + slot: GraphQLTypes["SlotComponentInput"] | null, + capacity: number | null, + comments: string | null, + ressourceModel: GraphQLTypes["BookingsRessourceEnum"], + ressourceId: string, + checkoutInfo: GraphQLTypes["CheckoutEngineInputI"] | null +}; + ["SlotComponentInput"]: { + startDate: GraphQLTypes["DateTime"], + endDate: GraphQLTypes["DateTime"], + startTime: string, + endTime: string +}; + ["CheckoutEngineInputI"]: { + contactInfo: GraphQLTypes["BuyerInfo"], + billingInfo: GraphQLTypes["BuyerInfo"], + token: string | null +}; + ["NewMessageInput"]: { + roomId: string, + message: string, + file: GraphQLTypes["ImageInput"] | null +}; + ["EditMessageInput"]: { + message: string +} + } +export enum AccountTypeEnum { + admin = "admin", + user = "user", + pro = "pro", + public = "public", + organisationOwner = "organisationOwner", + storeManager = "storeManager" +} +export enum AccountGenderEnum { + m = "m", + f = "f", + other = "other" +} +export enum AvailableTranslation { + en = "en", + fr = "fr", + nl = "nl", + de = "de" +} +export enum SettingsTypeEnum { + notifications = "notifications", + promotions = "promotions", + reminders = "reminders" +} +export enum cardTypeEnum { + alipay = "alipay", + au_becs_debit = "au_becs_debit", + bacs_debit = "bacs_debit", + bancontact = "bancontact", + card = "card", + eps = "eps", + fpx = "fpx", + giropay = "giropay", + ideal = "ideal", + p24 = "p24", + sepa_debit = "sepa_debit", + sofort = "sofort" +} +export enum cardBrandEnum { + amex = "amex", + diners = "diners", + discover = "discover", + jcb = "jcb", + mastercard = "mastercard", + unionpay = "unionpay", + visa = "visa", + unknown = "unknown" +} +export enum RessourceEnum { + workspaces = "workspaces" +} +export enum CountryCodesComponentEnum { + AF = "AF", + AX = "AX", + AL = "AL", + DZ = "DZ", + AS = "AS", + AD = "AD", + AO = "AO", + AI = "AI", + AQ = "AQ", + AG = "AG", + AR = "AR", + AM = "AM", + AW = "AW", + AU = "AU", + AT = "AT", + AZ = "AZ", + BS = "BS", + BH = "BH", + BD = "BD", + BB = "BB", + BY = "BY", + BE = "BE", + BZ = "BZ", + BJ = "BJ", + BM = "BM", + BT = "BT", + BO = "BO", + BA = "BA", + BW = "BW", + BV = "BV", + BR = "BR", + IO = "IO", + BN = "BN", + BG = "BG", + BF = "BF", + BI = "BI", + KH = "KH", + CM = "CM", + CA = "CA", + CV = "CV", + KY = "KY", + CF = "CF", + TD = "TD", + CL = "CL", + CN = "CN", + CX = "CX", + CC = "CC", + CO = "CO", + KM = "KM", + CG = "CG", + CD = "CD", + CK = "CK", + CR = "CR", + CI = "CI", + HR = "HR", + CU = "CU", + CY = "CY", + CZ = "CZ", + DK = "DK", + DJ = "DJ", + DM = "DM", + DO = "DO", + EC = "EC", + EG = "EG", + SV = "SV", + GQ = "GQ", + ER = "ER", + EE = "EE", + ET = "ET", + FK = "FK", + FO = "FO", + FJ = "FJ", + FI = "FI", + FR = "FR", + GF = "GF", + PF = "PF", + TF = "TF", + GA = "GA", + GM = "GM", + GE = "GE", + DE = "DE", + GH = "GH", + GI = "GI", + GR = "GR", + GL = "GL", + GD = "GD", + GP = "GP", + GU = "GU", + GT = "GT", + GG = "GG", + GN = "GN", + GW = "GW", + GY = "GY", + HT = "HT", + HM = "HM", + VA = "VA", + HN = "HN", + HK = "HK", + HU = "HU", + IS = "IS", + IN = "IN", + ID = "ID", + IR = "IR", + IQ = "IQ", + IE = "IE", + IM = "IM", + IL = "IL", + IT = "IT", + JM = "JM", + JP = "JP", + JE = "JE", + JO = "JO", + KZ = "KZ", + KE = "KE", + KI = "KI", + KR = "KR", + KW = "KW", + KG = "KG", + LA = "LA", + LV = "LV", + LB = "LB", + LS = "LS", + LR = "LR", + LY = "LY", + LI = "LI", + LT = "LT", + LU = "LU", + MO = "MO", + MK = "MK", + MG = "MG", + MW = "MW", + MY = "MY", + MV = "MV", + ML = "ML", + MT = "MT", + MH = "MH", + MQ = "MQ", + MR = "MR", + MU = "MU", + YT = "YT", + MX = "MX", + FM = "FM", + MD = "MD", + MC = "MC", + MN = "MN", + ME = "ME", + MS = "MS", + MA = "MA", + MZ = "MZ", + MM = "MM", + NA = "NA", + NR = "NR", + NP = "NP", + NL = "NL", + AN = "AN", + NC = "NC", + NZ = "NZ", + NI = "NI", + NE = "NE", + NG = "NG", + NU = "NU", + NF = "NF", + MP = "MP", + NO = "NO", + OM = "OM", + PK = "PK", + PW = "PW", + PS = "PS", + PA = "PA", + PG = "PG", + PY = "PY", + PE = "PE", + PH = "PH", + PN = "PN", + PL = "PL", + PT = "PT", + PR = "PR", + QA = "QA", + RE = "RE", + RO = "RO", + RU = "RU", + RW = "RW", + BL = "BL", + SH = "SH", + KN = "KN", + LC = "LC", + MF = "MF", + PM = "PM", + VC = "VC", + WS = "WS", + SM = "SM", + ST = "ST", + SA = "SA", + SN = "SN", + RS = "RS", + SC = "SC", + SL = "SL", + SG = "SG", + SK = "SK", + SI = "SI", + SB = "SB", + SO = "SO", + ZA = "ZA", + GS = "GS", + ES = "ES", + LK = "LK", + SD = "SD", + SR = "SR", + SJ = "SJ", + SZ = "SZ", + SE = "SE", + CH = "CH", + SY = "SY", + TW = "TW", + TJ = "TJ", + TZ = "TZ", + TH = "TH", + TL = "TL", + TG = "TG", + TK = "TK", + TO = "TO", + TT = "TT", + TN = "TN", + TR = "TR", + TM = "TM", + TC = "TC", + TV = "TV", + UG = "UG", + UA = "UA", + AE = "AE", + GB = "GB", + US = "US", + UM = "UM", + UY = "UY", + UZ = "UZ", + VU = "VU", + VE = "VE", + VN = "VN", + VG = "VG", + VI = "VI", + WF = "WF", + EH = "EH", + YE = "YE", + ZM = "ZM", + ZW = "ZW", + XK = "XK", + KP = "KP", + SS = "SS", + SX = "SX", + BQ = "BQ", + CW = "CW" +} +export enum ListEnum { + equipment = "equipment", + feature = "feature" +} +export enum WorkspaceTypeEnum { + meeting = "meeting", + individual = "individual", + open = "open" +} +export enum AvailableCurrency { + eur = "eur", + usd = "usd" +} +export enum ModelLoadersEnum { + streams = "streams", + accounts = "accounts", + workspaces = "workspaces", + bookings = "bookings" +} +export enum DurationTypeEnum { + startToEnd = "startToEnd", + slot = "slot" +} +export enum StatusEnum { + draft = "draft", + processing = "processing", + validated = "validated", + cancelled = "cancelled", + expired = "expired" +} +export enum BookingsRessourceEnum { + workspaces = "workspaces" +} +export enum RangeType { + strict = "strict", + intersect = "intersect", + intersectLarge = "intersectLarge", + included = "included" +} +export enum DateTabType { + today = "today", + tomorrow = "tomorrow", + upcomming = "upcomming", + past = "past" +} +export enum ProductRessourceEnum { + workspaces = "workspaces" +} +export enum CurrencyEnum { + eur = "eur", + usd = "usd" +} +export enum InvoicingProvider { + directInvoice = "directInvoice" +} +export enum OrderBusinessStatusEnum { + processing = "processing", + shipped = "shipped", + completed = "completed", + canceled = "canceled", + refund = "refund" +} +export enum OrderStatusEnum { + draft = "draft", + processing = "processing", + shipped = "shipped", + completed = "completed", + canceled = "canceled", + refund = "refund" +} +export enum PaymentProvider { + free = "free", + stripe = "stripe", + bancontact = "bancontact", + bankWire = "bankWire" +} +export enum PaymentStatusEnum { + pending = "pending", + actionNeeded = "actionNeeded", + paid = "paid", + free = "free" +} +export enum PromoType { + fixed = "fixed", + percentage = "percentage" +} +export class GraphQLError extends Error { + constructor(public response: GraphQLResponse) { + super(""); + console.error(response); + } + toString() { + return "GraphQL Response Error"; + } + } + + +export type UnwrapPromise = T extends Promise ? R : T; +export type ZeusState Promise> = NonNullable< + UnwrapPromise> +>; +export type ZeusHook< + T extends ( + ...args: any[] + ) => Record Promise>, + N extends keyof ReturnType +> = ZeusState[N]>; + +type WithTypeNameValue = T & { + __typename?: true; +}; +type AliasType = WithTypeNameValue & { + __alias?: Record>; +}; +export interface GraphQLResponse { + data?: Record; + errors?: Array<{ + message: string; + }>; +} +type DeepAnify = { + [P in keyof T]?: any; +}; +type IsPayLoad = T extends [any, infer PayLoad] ? PayLoad : T; +type IsArray = T extends Array ? InputType[] : InputType; +type FlattenArray = T extends Array ? R : T; + +type NotUnionTypes, DST> = { + [P in keyof DST]: SRC[P] extends '__union' & infer R ? never : P; +}[keyof DST]; + +type ExtractUnions, DST> = { + [P in keyof SRC]: SRC[P] extends '__union' & infer R + ? P extends keyof DST + ? IsArray + : {} + : never; +}[keyof SRC]; + +type IsInterfaced, DST> = FlattenArray extends ZEUS_INTERFACES | ZEUS_UNIONS + ? ExtractUnions & + { + [P in keyof Omit>, '__typename'>]: DST[P] extends true + ? SRC[P] + : IsArray; + } + : { + [P in keyof Pick]: DST[P] extends true ? SRC[P] : IsArray; + }; + + + +export type MapType = SRC extends DeepAnify ? IsInterfaced : never; +type InputType = IsPayLoad extends { __alias: infer R } + ? { + [P in keyof R]: MapType; + } & + MapType, '__alias'>> + : MapType>; +type Func

= (...args: P) => R; +type AnyFunc = Func; +export type ArgsType = F extends Func ? P : never; +export type OperationToGraphQL = (o: Z | V, variables?: Record) => Promise>; +export type SubscriptionToGraphQL = ( + o: Z | V, + variables?: Record, +) => { + ws: WebSocket; + on: (fn: (args: InputType) => void) => void; + off: (e: { data?: InputType; code?: number; reason?: string; message?: string }) => void; + error: (e: { data?: InputType; message?: string }) => void; + open: () => void; +}; +export type CastToGraphQL = (resultOfYourQuery: any) => (o: Z | V) => InputType; +export type SelectionFunction = (t: T | V) => T; +export type fetchOptions = ArgsType; +type websocketOptions = typeof WebSocket extends new ( + ...args: infer R +) => WebSocket + ? R + : never; +export type chainOptions = + | [fetchOptions[0], fetchOptions[1] & {websocket?: websocketOptions}] + | [fetchOptions[0]]; +export type FetchFunction = ( + query: string, + variables?: Record, +) => Promise; +export type SubscriptionFunction = ( + query: string, + variables?: Record, +) => void; +type NotUndefined = T extends undefined ? never : T; +export type ResolverType = NotUndefined; + + + +export const ZeusSelect = () => ((t: any) => t) as SelectionFunction; + +export const ScalarResolver = (scalar: string, value: any) => { + switch (scalar) { + case 'String': + return `${JSON.stringify(value)}`; + case 'Int': + return `${value}`; + case 'Float': + return `${value}`; + case 'Boolean': + return `${value}`; + case 'ID': + return `"${value}"`; + case 'enum': + return `${value}`; + case 'scalar': + return `${value}`; + default: + return false; + } +}; + + +export const TypesPropsResolver = ({ + value, + type, + name, + key, + blockArrays +}: { + value: any; + type: string; + name: string; + key?: string; + blockArrays?: boolean; +}): string => { + if (value === null) { + return `null`; + } + let resolvedValue = AllTypesProps[type][name]; + if (key) { + resolvedValue = resolvedValue[key]; + } + if (!resolvedValue) { + throw new Error(`Cannot resolve ${type} ${name}${key ? ` ${key}` : ''}`) + } + const typeResolved = resolvedValue.type; + const isArray = resolvedValue.array; + const isArrayRequired = resolvedValue.arrayRequired; + if (typeof value === 'string' && value.startsWith(`ZEUS_VAR$`)) { + const isRequired = resolvedValue.required ? '!' : ''; + let t = `${typeResolved}`; + if (isArray) { + if (isRequired) { + t = `${t}!`; + } + t = `[${t}]`; + if(isArrayRequired){ + t = `${t}!`; + } + }else{ + if (isRequired) { + t = `${t}!`; + } + } + return `\$${value.split(`ZEUS_VAR$`)[1]}__ZEUS_VAR__${t}`; + } + if (isArray && !blockArrays) { + return `[${value + .map((v: any) => TypesPropsResolver({ value: v, type, name, key, blockArrays: true })) + .join(',')}]`; + } + const reslovedScalar = ScalarResolver(typeResolved, value); + if (!reslovedScalar) { + const resolvedType = AllTypesProps[typeResolved]; + if (typeof resolvedType === 'object') { + const argsKeys = Object.keys(resolvedType); + return `{${argsKeys + .filter((ak) => value[ak] !== undefined) + .map( + (ak) => `${ak}:${TypesPropsResolver({ value: value[ak], type: typeResolved, name: ak })}` + )}}`; + } + return ScalarResolver(AllTypesProps[typeResolved], value) as string; + } + return reslovedScalar; +}; + + +const isArrayFunction = ( + parent: string[], + a: any[] +) => { + const [values, r] = a; + const [mainKey, key, ...keys] = parent; + const keyValues = Object.keys(values).filter((k) => typeof values[k] !== 'undefined'); + + if (!keys.length) { + return keyValues.length > 0 + ? `(${keyValues + .map( + (v) => + `${v}:${TypesPropsResolver({ + value: values[v], + type: mainKey, + name: key, + key: v + })}` + ) + .join(',')})${r ? traverseToSeekArrays(parent, r) : ''}` + : traverseToSeekArrays(parent, r); + } + + const [typeResolverKey] = keys.splice(keys.length - 1, 1); + let valueToResolve = ReturnTypes[mainKey][key]; + for (const k of keys) { + valueToResolve = ReturnTypes[valueToResolve][k]; + } + + const argumentString = + keyValues.length > 0 + ? `(${keyValues + .map( + (v) => + `${v}:${TypesPropsResolver({ + value: values[v], + type: valueToResolve, + name: typeResolverKey, + key: v + })}` + ) + .join(',')})${r ? traverseToSeekArrays(parent, r) : ''}` + : traverseToSeekArrays(parent, r); + return argumentString; +}; + + +const resolveKV = (k: string, v: boolean | string | { [x: string]: boolean | string }) => + typeof v === 'boolean' ? k : typeof v === 'object' ? `${k}{${objectToTree(v)}}` : `${k}${v}`; + + +const objectToTree = (o: { [x: string]: boolean | string }): string => + `{${Object.keys(o).map((k) => `${resolveKV(k, o[k])}`).join(' ')}}`; + + +const traverseToSeekArrays = (parent: string[], a?: any): string => { + if (!a) return ''; + if (Object.keys(a).length === 0) { + return ''; + } + let b: Record = {}; + if (Array.isArray(a)) { + return isArrayFunction([...parent], a); + } else { + if (typeof a === 'object') { + Object.keys(a) + .filter((k) => typeof a[k] !== 'undefined') + .map((k) => { + if (k === '__alias') { + Object.keys(a[k]).map((aliasKey) => { + const aliasOperations = a[k][aliasKey]; + const aliasOperationName = Object.keys(aliasOperations)[0]; + const aliasOperation = aliasOperations[aliasOperationName]; + b[ + `${aliasOperationName}__alias__${aliasKey}: ${aliasOperationName}` + ] = traverseToSeekArrays([...parent, aliasOperationName], aliasOperation); + }); + } else { + b[k] = traverseToSeekArrays([...parent, k], a[k]); + } + }); + } else { + return ''; + } + } + return objectToTree(b); +}; + + +const buildQuery = (type: string, a?: Record) => + traverseToSeekArrays([type], a); + + +const inspectVariables = (query: string) => { + const regex = /\$\b\w*__ZEUS_VAR__\[?[^!^\]^\s^,^\)^\}]*[!]?[\]]?[!]?/g; + let result; + const AllVariables: string[] = []; + while ((result = regex.exec(query))) { + if (AllVariables.includes(result[0])) { + continue; + } + AllVariables.push(result[0]); + } + if (!AllVariables.length) { + return query; + } + let filteredQuery = query; + AllVariables.forEach((variable) => { + while (filteredQuery.includes(variable)) { + filteredQuery = filteredQuery.replace(variable, variable.split('__ZEUS_VAR__')[0]); + } + }); + return `(${AllVariables.map((a) => a.split('__ZEUS_VAR__')) + .map(([variableName, variableType]) => `${variableName}:${variableType}`) + .join(', ')})${filteredQuery}`; +}; + + +export const queryConstruct = (t: 'query' | 'mutation' | 'subscription', tName: string) => (o: Record) => + `${t.toLowerCase()}${inspectVariables(buildQuery(tName, o))}`; + + +const fullChainConstruct = (fn: FetchFunction) => (t: 'query' | 'mutation' | 'subscription', tName: string) => ( + o: Record, + variables?: Record, +) => fn(queryConstruct(t, tName)(o), variables).then((r:any) => { + seekForAliases(r) + return r +}); + +export const fullChainConstructor = ( + fn: F, + operation: 'query' | 'mutation' | 'subscription', + key: R, +) => + ((o, variables) => fullChainConstruct(fn)(operation, key)(o as any, variables)) as OperationToGraphQL< + ValueTypes[R], + GraphQLTypes[R] + >; + + +const fullSubscriptionConstruct = (fn: SubscriptionFunction) => ( + t: 'query' | 'mutation' | 'subscription', + tName: string, +) => (o: Record, variables?: Record) => + fn(queryConstruct(t, tName)(o), variables); + +export const fullSubscriptionConstructor = ( + fn: F, + operation: 'query' | 'mutation' | 'subscription', + key: R, +) => + ((o, variables) => fullSubscriptionConstruct(fn)(operation, key)(o as any, variables)) as SubscriptionToGraphQL< + ValueTypes[R], + GraphQLTypes[R] + >; + + +const seekForAliases = (response: any) => { + const traverseAlias = (value: any) => { + if (Array.isArray(value)) { + value.forEach(seekForAliases); + } else { + if (typeof value === 'object') { + seekForAliases(value); + } + } + }; + if (typeof response === 'object' && response) { + const keys = Object.keys(response); + if (keys.length < 1) { + return; + } + keys.forEach((k) => { + const value = response[k]; + if (k.indexOf('__alias__') !== -1) { + const [operation, alias] = k.split('__alias__'); + response[alias] = { + [operation]: value, + }; + delete response[k]; + } + traverseAlias(value); + }); + } +}; + + +export const $ = (t: TemplateStringsArray): any => `ZEUS_VAR$${t.join('')}`; + + +export const resolverFor = < + T extends keyof ValueTypes, + Z extends keyof ValueTypes[T], + Y extends ( + args: Required[Z] extends [infer Input, any] ? Input : any, + source: any, + ) => Z extends keyof ModelTypes[T] ? ModelTypes[T][Z] | Promise : any +>( + type: T, + field: Z, + fn: Y, +) => fn as (args?: any,source?: any) => any; + + +const handleFetchResponse = ( + response: Parameters['then']>[0], Function>>[0] +): Promise => { + if (!response.ok) { + return new Promise((_, reject) => { + response.text().then(text => { + try { reject(JSON.parse(text)); } + catch (err) { reject(text); } + }).catch(reject); + }); + } + return response.json(); +}; + +export const apiFetch = (options: fetchOptions) => (query: string, variables: Record = {}) => { + let fetchFunction = fetch; + let queryString = query; + let fetchOptions = options[1] || {}; + if (fetchOptions.method && fetchOptions.method === 'GET') { + queryString = encodeURIComponent(query); + return fetchFunction(`${options[0]}?query=${queryString}`, fetchOptions) + .then(handleFetchResponse) + .then((response: GraphQLResponse) => { + if (response.errors) { + throw new GraphQLError(response); + } + return response.data; + }); + } + return fetchFunction(`${options[0]}`, { + body: JSON.stringify({ query: queryString, variables }), + method: 'POST', + headers: { + 'Content-Type': 'application/json' + }, + ...fetchOptions + }) + .then(handleFetchResponse) + .then((response: GraphQLResponse) => { + if (response.errors) { + throw new GraphQLError(response); + } + return response.data; + }); + }; + + +export const apiSubscription = (options: chainOptions) => ( + query: string, + variables: Record = {}, + ) => { + try { + const queryString = options[0] + '?query=' + encodeURIComponent(query); + const wsString = queryString.replace('http', 'ws'); + const host = (options.length > 1 && options[1]?.websocket?.[0]) || wsString; + const webSocketOptions = options[1]?.websocket || [host]; + const ws = new WebSocket(...webSocketOptions); + return { + ws, + on: (e: (args: any) => void) => { + ws.onmessage = (event:any) => { + if(event.data){ + const parsed = JSON.parse(event.data) + const data = parsed.data + if (data) { + seekForAliases(data); + } + return e(data); + } + }; + }, + off: (e: (args: any) => void) => { + ws.onclose = e; + }, + error: (e: (args: any) => void) => { + ws.onerror = e; + }, + open: (e: () => void) => { + ws.onopen = e; + }, + }; + } catch { + throw new Error('No websockets implemented'); + } + }; + + +export const Thunder = (fn: FetchFunction, subscriptionFn: SubscriptionFunction) => ({ + query: fullChainConstructor(fn,'query', 'Query'), +mutation: fullChainConstructor(fn,'mutation', 'Mutation') +}); + +export const Chain = (...options: chainOptions) => ({ + query: fullChainConstructor(apiFetch(options),'query', 'Query'), +mutation: fullChainConstructor(apiFetch(options),'mutation', 'Mutation') +}); +export const Zeus = { + query: (o:ValueTypes["Query"]) => queryConstruct('query', 'Query')(o), +mutation: (o:ValueTypes["Mutation"]) => queryConstruct('mutation', 'Mutation')(o) +}; +export const Cast = { + query: ((o: any) => (_: any) => o) as CastToGraphQL< + ValueTypes["Query"], + GraphQLTypes["Query"] +>, +mutation: ((o: any) => (_: any) => o) as CastToGraphQL< + ValueTypes["Mutation"], + GraphQLTypes["Mutation"] +> +}; +export const Selectors = { + query: ZeusSelect(), +mutation: ZeusSelect() +}; + + +export const Gql = Chain('http://localhost:4000') \ No newline at end of file diff --git a/__tests/app/index.ts b/__tests/app/index.ts new file mode 100644 index 0000000..2c2ce53 --- /dev/null +++ b/__tests/app/index.ts @@ -0,0 +1,16 @@ +import { finaliseCheckoutWithStripe, reserveOneBooking } from './src/bookings'; +import { getOneWorkspace, searchWorkspace } from './src/workspace'; + +/* + Workspace +*/ + +// searchWorkspace(); +// getOneWorkspace(); + +/* + Bookings +*/ + +reserveOneBooking(); +// finaliseCheckoutWithStripe(); diff --git a/__tests/app/src/bookings.ts b/__tests/app/src/bookings.ts new file mode 100644 index 0000000..c5c8155 --- /dev/null +++ b/__tests/app/src/bookings.ts @@ -0,0 +1,72 @@ +import { thunder } from '../config'; +import { BookingsRessourceEnum, DurationTypeEnum } from '../helpers/graphql-zeus'; + +export const reserveOneBooking = async () => { + const reserveOneBooking = await thunder.mutation({ + reserveOneBooking: [ + { + input: { + durationType: DurationTypeEnum.slot, + slot: { + startDate: '2021-03-18T23:00:00.000Z', + endDate: '2021-03-19T23:00:00.000Z', + startTime: '12:00', + endTime: '14:30', + }, + ressourceId: '6187782d-0445-4c0d-8c31-6c0c48a95312', + ressourceModel: BookingsRessourceEnum.workspaces, + }, + }, + { + _id: true, + 'finalPrice':true, + subTotalPrice:true, + "vatPrice":true, + paymentIntent: { + stripePaymentIntentData: { + client_secret: true, + id: true, + }, + }, + }, + ], + }); +}; + +export const finaliseCheckoutWithStripe = async () => { + return await thunder.mutation({ + finaliseCheckoutWithStripe: [ + { + input: { + billingInfo: { + firstName: 'Jay', + lastName: 'Son', + email: '', + address: { + city: 'BX', + country: 'Belgium', + }, + }, + contactInfo: { + firstName: 'Jay', + lastName: 'Son', + email: '', + address: { + city: 'BX', + country: 'Belgium', + }, + }, + }, + }, + { + _id: true, + paymentIntent: { + stripePaymentIntentData: { + client_secret: true, + id: true, + }, + }, + }, + ], + }); +}; diff --git a/__tests/app/src/workspace.ts b/__tests/app/src/workspace.ts new file mode 100644 index 0000000..0ab9d3f --- /dev/null +++ b/__tests/app/src/workspace.ts @@ -0,0 +1,49 @@ +import { thunder } from '../config'; +import { BookingsRessourceEnum, DurationTypeEnum } from '../helpers/graphql-zeus'; + +export const searchWorkspace = async () => { + const searchWorkspace = await thunder.query({ + workspacesSearchMany: [ + { + dateFrom: '2021-03-18T00:00:00', + dateTo: '2021-03-20T00:00:00', + hoursFrom: '10:00', + hoursTo: '12:00', + }, + { + _id: true, + title: true, + }, + ], + }); + + console.log(searchWorkspace); + return searchWorkspace; +}; + +export const getOneWorkspace = async () => { + return await thunder.query({ + workspacesGetOne: [ + { + id: '6187782d-0445-4c0d-8c31-6c0c48a95312', + }, + { + _id: true, + title: true, + getOwner: { + firstName: true, + lastName: true, + }, + getBookings: [ + {}, + { + dates: { + startDate: true, + endDate: true, + }, + }, + ], + }, + ], + }); +}; diff --git a/__tests__/booking.gql b/__tests__/booking.gql new file mode 100644 index 0000000..725e530 --- /dev/null +++ b/__tests__/booking.gql @@ -0,0 +1,11 @@ +mutation createBooking { + createOneBooking(input:{ressourceModel:workspaces,ressourceId:"718aa94e-f13d-4ee3-9610-c12c98750c08",dates:{startDate:"2020-11-01T23:15:17.044+0000",endDate:"2020-11-02T23:15:17.044+0000"},capacity:10}){_id,ownerId,owner{email}} +} + +query bookingsGetMany { + bookingsGetMany{_id, workspace{_id,title}} +} + +query bookingsGetOne { + bookingsGetOne(id:"2a3c6b3d-e8e5-4a4d-9c10-6e438ac22e10"){_id, workspace{_id,title}} +} \ No newline at end of file diff --git a/__tests__/signin.gql b/__tests__/signin.gql new file mode 100644 index 0000000..22c01b5 --- /dev/null +++ b/__tests__/signin.gql @@ -0,0 +1,7 @@ + + +mutation resetPassword { + resetPassword(input: { email: "sanawar@makeit-studio.com" }) { + message + } +} \ No newline at end of file diff --git a/__tests__/signup.gql b/__tests__/signup.gql new file mode 100644 index 0000000..730158e --- /dev/null +++ b/__tests__/signup.gql @@ -0,0 +1,2 @@ +query me {me{paymentInfo{stripeInfo{customerId},paymentMethods{sPayMethodId,default,nameOnCard},billingInfos{firstName,lastName,vatNumber}}}} +query login {login(creds:{email:"nicolas.a.bernier@gmail.com",password:"aaaaaa"}){localId,idToken}} diff --git a/__tests__/workspace.gql b/__tests__/workspace.gql new file mode 100644 index 0000000..d46aec7 --- /dev/null +++ b/__tests__/workspace.gql @@ -0,0 +1,66 @@ +query workspacesSearchMany { + workspacesSearchMany( + workspaceType: open + priceMin: 5 + priceMax: 15 + dateFrom: "2020-09-25T15:58:40.803+0000" + dateTo: "2020-12-20T15:58:40.803+0000" + hoursFrom: "08:00" + hoursTo: "12:00" + ) { + _id + title { + en + } + } +} + +query workspaceGetOne { + workspacesGetOne(id: "5061682a-11df-48a8-9894-9ebc577c63d5") { + _id + title { + en + } + } +} + +query workspacesGetMine { + workspacesGetMine { + _id + title { + en + } + } +} +mutation workspacesAddOne { + workspacesAddOne( + input: { + place: { + placeId: "ChIJrZYyJf_Rw0cR8I227YjcDEQ" + loc: { type: "Point", coordinates: [4.3879295, 50.6819832] } + formattedAddress: "Avenue de la Paix 36, 1420 Braine-l'Alleud, Belgium" + } + workspaceType: open + pricingPerHour: 25 + currency: eur + maxCapacity: 4 + minStay: 1 + maxStay: 2 + title: { fr: "Le rouge petillant", en: "The Red Velvet" } + availability: { + dateFrom: "2020-09-24T15:58:40.803+0000" + dateTo: "2020-12-24T15:58:40.803+0000" + weekend: true + hours: [{ from: "08:00", to: "20:00" }] + exceptions: [{ dateFrom: "2020-10-24T15:58:40.803+0000", dateTo: "2020-10-31T15:58:40.803+0000" }] + } + equipmentIds: [] + featureIds: [] + } + ) { + _id + title { + en + } + } +} diff --git a/config/config.ts b/config/config.ts new file mode 100644 index 0000000..2186273 --- /dev/null +++ b/config/config.ts @@ -0,0 +1,24 @@ +import { seedErrorConfig } from '@seed/interfaces/components.errors'; +import { engineBookingErrorConfig } from '@services/module-booking/components.errors'; +import { enginePaymentErrorConfig } from '@services/module-payments/components/components.errors'; + +export enum NotificationEnum { + // emailCodeSignin = "emailCodeSignin", + // magicLink = "magicLink", + adminAccountAddOne = "adminAccountAddOne", + resetPassword = "resetPassword", + bookingUserConfirm = "bookingUserConfirm", + bookingProConfirm = "bookingProConfirm", + bookingUserCancel = "bookingUserCancel", + bookingProCancel = "bookingProCancel", + welcomeUser = "welcomeUser", +} + +export const ErrorsConfig = { + ...seedErrorConfig, + ...enginePaymentErrorConfig, + ...engineBookingErrorConfig, + + // Auth & Registration + 4000: 'You already completed your profile', +}; diff --git a/config/domains.yaml b/config/domains.yaml new file mode 100644 index 0000000..758fa1a --- /dev/null +++ b/config/domains.yaml @@ -0,0 +1,19 @@ +## Instal ## +## run : ./node_modules/.bin/sls plugin install --name serverless-domain-manager --config ./lib/seed/serverless-domains.yaml +## run : SLS_DEBUG=* ./node_modules/.bin/sls create_domain --config ./lib/seed/serverless-domains.yaml --aws-profile awsProfile + +## SLS_DEBUG=* ./node_modules/.bin/sls delete_domain --config ./lib/seed/serverless-domains.yaml --aws-profile awsProfile +domains: + production: api.work-in-flex.com + staging: staging-api.work-in-flex.com + dev: dev-api.work-in-flex.com + dev-v2: dev-v2-api.work-in-flex.com +customDomain: + basePath: '${self:service}' + domainName: ${self:custom.domains.${self:custom.stage}} + stage: '${self:custom.stage}' + createRoute53Record: true + certificateName: work-in-flex.com + certificateArn: arn:aws:acm:us-east-1:555646219416:certificate/07e83105-1554-4e3d-b968-0a34d77911ad + endpointType: 'edge' + securityPolicy: tls_1_2 diff --git a/lib/__cronjobs/serverless-cron.yaml b/lib/__cronjobs/serverless-cron.yaml new file mode 100644 index 0000000..e21f185 --- /dev/null +++ b/lib/__cronjobs/serverless-cron.yaml @@ -0,0 +1,32 @@ +app: ${file(./package.json):name} +service: cronjob +package: + individually: false +provider: + name: aws + stage: ${opt:stage,'dev'} + runtime: nodejs12.x + environment: + AWS_compute: ${self:service}-${self:provider.stage}-compute + iamRoleStatements: + - Effect: Allow + Action: + - lambda:InvokeFunction + Resource: '*' +plugins: + - serverless-webpack +custom: + stage: ${opt:stage, self:provider.stage} + webpack: + webpackConfig: './lib/seed/webpack.config.js' # Name of webpack configuration file + includeModules: + forceExclude: + - aws-sdk + - puppeteer + +functions: + removeDraftBookings: + handler: lib/__cronjobs/handler.removeDraftBookings + timeout: 300 # optional, in seconds, default is 6 + events: + - schedule: rate(1 minute) diff --git a/lib/__hooks/index.ts b/lib/__hooks/index.ts new file mode 100644 index 0000000..0c23f16 --- /dev/null +++ b/lib/__hooks/index.ts @@ -0,0 +1,29 @@ +export class AsyncHooksService { + /* [MODULES - SHOP] */ + + public async afterCheckout(): Promise { + console.log('[NO ACTION] - afterCheckout'); + } + + public async afterOrderCancel(data,ctx): Promise { + console.log('[NO ACTION] - afterCheckout'); + } + + public async afterOrdersMarkOneAsPaid(data,ctx): Promise { + console.log('[NO ACTION] - afterCheckout'); + } + + public async afterOrdersMarkOneAsUnPaid(data,ctx): Promise { + console.log('[NO ACTION] - afterCheckout'); + } + + public async afterOrdersReimbursed(data,ctx): Promise { + console.log('[NO ACTION] - afterOrdersReimbursed'); + } + + /* [MODULES - EVENTS] */ + + public async afterCheckoutEvent(): Promise { + console.log('[NO ACTION] - afterCheckout'); + } +} diff --git a/lib/seed/.gitignore b/lib/seed/.gitignore new file mode 100644 index 0000000..3c49c37 --- /dev/null +++ b/lib/seed/.gitignore @@ -0,0 +1,19 @@ +node_modules +.DS_Store +package-lock.json + +.build +.webpack + +*/**/.env +*/**/postman.json +*/**/schema.d.ts +*/**/*.d.ts + +src/config/.env.json + +.serverless/ + +config/.env.json + +serverless.yaml diff --git a/lib/seed/README.md b/lib/seed/README.md new file mode 100644 index 0000000..b9eaf43 --- /dev/null +++ b/lib/seed/README.md @@ -0,0 +1,73 @@ +# Make-it Seed V5 + +[![](https://ci6.googleusercontent.com/proxy/2HHmNs0zn0uGJb9RqML73pX6Bmd-BntMaIYR6IpXeIHKqn2_tG60C1pLNtMHoLHOk1I5CT8k5x9gvb-C1zuudHnD5HDOpvHljVYcKmDL--rb0Ar8IT4UB7YaG1c=s0-d-e1-ft#https://makeit-assets.s3.eu-central-1.amazonaws.com/signature_sanawar.png)](https://makeit-studio.com) + +![Build Status](https://travis-ci.org/joemccann/dillinger.svg?branch=master) + +### GrapQL carefull + +- Fields resolvers must await the response (cannot return a promise) +- Classical errors + -- Cannot determine GraphQL output type for 'xxx' of 'YOUR_CLASS' class. Does the value used as its TS type or explicit type is decorated with a proper decorator or is it a proper output value? + --- You forgot to put @ObjectType or @InputType in your class definition. Or your forgot to precise on the decorator what type of field it was @Field(() => YOUR_CLASS) + +### Serverless carefull + +- Test your serverless file npx sls print --config ./devOps/serverless-upload.yaml + +### Installation + +In order to make the seed run and setup your project you need to + +- copy the account service in your application folder (see example folder) +- copy the \_\_loaders file in your application folder (see example folder) +- copy the \_\_resolvers.ts file in your application folder (see example folder) + +- copy the .env file in your config folder (see example folder) + - put the correct admin email + +### Domains + +- run : ./node_modules/.bin/sls plugin install --name serverless-domain-manager --config ./lib/seed/serverless-domains.yaml +- run : SLS_DEBUG=\* ./node_modules/.bin/sls create_domain --config ./lib/seed/serverless-domains.yaml --aws-profile awsProfile + +- to delete : SLS_DEBUG=\* ./node_modules/.bin/sls delete_domain --config ./lib/seed/serverless-domains.yaml --aws-profile awsProfile + +### VSCODE CONFIG + +npm i -g eslint eslint-config-prettier eslint-plugin-prettier prettier + +### If you need REST APIS + +npm install --save express serverless-http + +Check examples/app-rest.ts + +# SOME ENV CONFIG + +TWILIO_FROM_NUMBER +TWILIO_ACCOUNT_SID +TWILIO_AUTH_TOKEN + +# How to write tests + +npm i -g graphql-zeus +Have your dev running +zeus http://localhost:4000 ./**tests/app/helpers --ts +zeus http://localhost:4001 ./**tests/admin/helpers --ts + +# Search Engine + +--- Free form text +(1) Create text index on searchT field + +# Cronjobs + +copy examples/**cronjobs to lib/**cronjobs + +Add this to your package.json +"cronjobs": [ +"./lib/__cronjobs", +"serverless-cron.yaml", +"handler.ts" +] diff --git a/lib/seed/__loaders.ts b/lib/seed/__loaders.ts new file mode 100644 index 0000000..a7c18cb --- /dev/null +++ b/lib/seed/__loaders.ts @@ -0,0 +1,46 @@ +import CategoryModel from '@services/module-cms/functions/categories/category.model'; + +import ListModel from '@services/module-cms/functions/lists/list.model'; +import PageModel from '@services/module-cms/functions/pages/pages.model'; + +import AccountModel from '@src/accounts/account.model'; +import DataLoader from 'dataloader'; +import StreamEngineModel from './engine/utils/streams/stream.model'; +import { buildEngineLoader, buildLoader } from './services/database/LoaderService'; +import NotificationModel from './services/notifications/notifications.model'; + +/* + + !!! Do not use, copy on parent only ! + +*/ +const seedModelLoaders = { + 'stream.changes': new StreamEngineModel(), + streams: new StreamEngineModel(), + notifications: new NotificationModel(), + accounts: new AccountModel(), + pages: new PageModel(), +}; +class SeedLoaders { + /* CMS */ + + listLoader: DataLoader; + categoryLoader: DataLoader; + // tagsLoader: DataLoader; + + // answerLoader: DataLoader; + + /* ACCOUNT */ + + accountLoader: DataLoader; + + public constructor() { + this.listLoader = buildLoader(new ListModel()); + this.categoryLoader = buildLoader(new CategoryModel()); + // this.tagsLoader = buildEngineLoader(new TagModel()); + + // this.answerLoader = buildEngineLoader(new AnswerEngineModel()); + + this.accountLoader = buildLoader(new AccountModel()); + } +} diff --git a/lib/seed/__tests/helper.ts b/lib/seed/__tests/helper.ts new file mode 100644 index 0000000..72e2f08 --- /dev/null +++ b/lib/seed/__tests/helper.ts @@ -0,0 +1,33 @@ +import { clog } from '../helpers/Utils'; +import fetch from 'node-fetch'; + +export function testRunQuery(url: string, token: string): (query: string, variables?: Record) => Promise { + // eslint-disable-next-line @typescript-eslint/explicit-function-return-type + return async (query, variables) => { + const headers: any = { + 'Content-Type': 'application/json', + }; + + if (token && (token as any) != '') headers.authorization = token; + + const r = JSON.stringify({ query, variables }); + clog({ query, variables }); + + const response = await fetch(url, { + body: r, + method: 'POST', + headers: headers, + }); + + try { + const text = await response.text(); + const res = JSON.parse(text); + if (res.errors) clog(res.errors[0]); + clog(res.data); + } catch (err) { + console.log(err); + } + + return; + }; +} diff --git a/lib/seed/app-admin.ts b/lib/seed/app-admin.ts new file mode 100644 index 0000000..945d036 --- /dev/null +++ b/lib/seed/app-admin.ts @@ -0,0 +1,66 @@ +import 'reflect-metadata'; +import { authMiddleware, errorMiddleware, ctxMiddleware, complexityMiddleware } from '@seed/graphql/Middleware'; + +import Firebase from '@seed/services/auth/FirebaseService'; + +import { AdminResolvers } from '@src/__indexes/__resolvers.index.admin'; +import { SettingsCache } from './graphql/Settings'; +import { buildSchema } from 'type-graphql'; + +// eslint-disable-next-line @typescript-eslint/no-var-requires +const { ApolloServer } = require('apollo-server'); + +const PORT = process.env.PORT || 4001; + +const bootstrap = async (): Promise => { + process.env.NODE_ENV = 'local'; + + const settingsI = SettingsCache.getInstance(); + await settingsI.refreshCache(); + + try { + const schema = await buildSchema({ + resolvers: AdminResolvers as any, + authChecker: authMiddleware, + }); + + const server = new ApolloServer({ + // typeDefs, + // resolvers, + schema, + formatError: errorMiddleware, + formatResponse: (response): any => { + return response; + }, + context: ctxMiddleware, + tracing: false, + plugins: [ + { + // eslint-disable-next-line @typescript-eslint/explicit-function-return-type + requestDidStart: () => ({ + // eslint-disable-next-line @typescript-eslint/explicit-function-return-type + didResolveOperation({ request, document }) { + return complexityMiddleware(schema, request.variables, document); + }, + }), + }, + ], + }); + + try { + const token = await Firebase.getInstance().createTokenId(process.env.TEST_ADMIN_EMAIL ? process.env.TEST_ADMIN_EMAIL : ''); + console.log(token); + } catch (error) { + console.log('no token available'); + } + + server.listen(PORT).then(({ url }) => { + console.log(`🚀 Server ready at ${url}`); + }); + } catch (error) { + console.log('[GRAPH ERROR]', error); + throw error; + } +}; + +bootstrap(); diff --git a/lib/seed/app.ts b/lib/seed/app.ts new file mode 100644 index 0000000..311e8da --- /dev/null +++ b/lib/seed/app.ts @@ -0,0 +1,72 @@ +import 'reflect-metadata'; +import { buildSchema, NonEmptyArray } from 'type-graphql'; + +import { authMiddleware, errorMiddleware, ctxMiddleware, complexityMiddleware } from '@seed/graphql/Middleware'; + +import Firebase from '@seed/services/auth/FirebaseService'; + +import { AppResolvers } from '@src/__indexes/__resolvers.index'; +import { SettingsCache } from './graphql/Settings'; +import { runBeforeAll } from '@src/__indexes/__bootstrap'; +import { nullToUndefined } from './helpers/Utils'; + +// eslint-disable-next-line @typescript-eslint/no-var-requires +const { ApolloServer } = require('apollo-server'); + +const PORT = process.env.PORT || 4000; + +const bootstrap = async (): Promise => { + process.env.NODE_ENV = 'local'; + + const settingsI = SettingsCache.getInstance(); + await settingsI.refreshCache(); + + try { + const schema = await buildSchema({ + resolvers: (AppResolvers as unknown) as NonEmptyArray, + authChecker: authMiddleware, + validate: false, // disable automatic validation or pass the default config object + }); + + const server = new ApolloServer({ + // typeDefs, + // resolvers, + schema, + formatError: errorMiddleware, + formatResponse: (response): any => { + return nullToUndefined(response); + }, + context: ctxMiddleware, + tracing: false, + plugins: [ + { + // eslint-disable-next-line @typescript-eslint/explicit-function-return-type + requestDidStart: () => ({ + // eslint-disable-next-line @typescript-eslint/explicit-function-return-type + didResolveOperation({ request, document }) { + return complexityMiddleware(schema, request.variables, document); + }, + }), + }, + ], + }); + + try { + const token = await Firebase.getInstance().createTokenId(process.env.TEST_USER_EMAIL ? process.env.TEST_USER_EMAIL : ''); + console.log(`Token for ${process.env.TEST_USER_EMAIL}`, token.idToken); + } catch (error) { + console.log('no token available'); + } + + await runBeforeAll(); + + server.listen(PORT).then(({ url }) => { + console.log(`🚀 Server ready at ${url}`); + }); + } catch (error) { + console.log('[GRAPH ERROR]', error); + throw error; + } +}; + +bootstrap(); diff --git a/lib/seed/devops/deploy-domains.js b/lib/seed/devops/deploy-domains.js new file mode 100644 index 0000000..2b563a5 --- /dev/null +++ b/lib/seed/devops/deploy-domains.js @@ -0,0 +1,117 @@ +/* eslint-disable @typescript-eslint/no-empty-function */ +/* eslint-disable @typescript-eslint/explicit-function-return-type */ +/* eslint-disable @typescript-eslint/no-var-requires */ +const execa = require('execa'); +const Listr = require('listr'); +const package = require('../../../package.json'); + +const fs = require('fs'); + +if (!package.devops) throw `No devops configuration found in package.json`; +if (!package.devops.services) throw `No devops services found in package.json`; +if (!package.devops.awsProfile) throw `No devops awsProfile found in package.json`; +if (!package.devops.awsRegion) throw 'No devops awsRegion found in package.json'; + +// Config input +const awsProfile = package.devops.awsProfile; +const awsRegion = package.devops.awsRegion; + +// Arguments checks + +const deployBase = function() { + return new Listr( + [ + { + title: 'Deploy domains for DEV', + task: async () => { + const subprocess = execa( + 'npx', + [ + './node_modules/.bin/sls', + 'create_domain', + '--config', + './lib/seed/devops/serverless-domains.yaml', + '--stage', + 'dev', + '--aws-profile', + awsProfile, + '--region', + awsRegion, + ], + { env: { 'SLS_DEBUG=*': '*' } }, + ); + + subprocess.stdout.pipe(fs.createWriteStream('.slslogs.txt', { flags: 'a' })); + + const { stdout } = await subprocess; + }, + }, + { + title: 'Deploy domains for STAGING', + task: async () => { + const subprocess = execa( + 'npx', + [ + './node_modules/.bin/sls', + 'create_domain', + '--config', + './lib/seed/devops/serverless-domains.yaml', + '--stage', + 'staging', + '--aws-profile', + awsProfile, + '--region', + awsRegion, + ], + { env: { 'SLS_DEBUG=*': '*' } }, + ); + + subprocess.stdout.pipe(fs.createWriteStream('.slslogs.txt', { flags: 'a' })); + + const { stdout } = await subprocess; + }, + }, + { + title: 'Deploy domains for PRODUCTION', + task: async () => { + const subprocess = execa( + 'npx', + [ + './node_modules/.bin/sls', + 'create_domain', + '--config', + './lib/seed/devops/serverless-domains.yaml', + '--stage', + 'production', + '--aws-profile', + awsProfile, + '--region', + awsRegion, + ], + { env: { 'SLS_DEBUG=*': '*' } }, + ); + + subprocess.stdout.pipe(fs.createWriteStream('.slslogs.txt', { flags: 'a' })); + + const { stdout } = await subprocess; + }, + }, + ], + { concurrent: false }, + ); +}; + +const tasksList = []; + +tasksList.push({ + title: 'Deploying Domains', + task: () => { + return deployBase(); + }, +}); + +const tasks = new Listr(tasksList, { concurrent: false }); + +tasks.run().catch((err) => { + console.error(err); +}); diff --git a/lib/seed/devops/deploy-v2.js b/lib/seed/devops/deploy-v2.js new file mode 100644 index 0000000..63001c2 --- /dev/null +++ b/lib/seed/devops/deploy-v2.js @@ -0,0 +1,98 @@ +/* eslint-disable @typescript-eslint/no-empty-function */ +/* eslint-disable @typescript-eslint/explicit-function-return-type */ +/* eslint-disable @typescript-eslint/no-var-requires */ +const execa = require('execa'); +const Listr = require('listr'); +const package = require('../../../package.json'); + +const fs = require('fs'); + +if (!package.devops) throw `No devops configuration found in package.json`; +if (!package.devops.services) throw `No devops services found in package.json`; +if (!package.devops.awsProfile) throw `No devops awsProfile found in package.json`; +if (!package.devops.awsRegion) throw 'No devops awsRegion found in package.json'; + +// Config input +const paths = package.devops.services; +const awsProfile = package.devops.awsProfile; +const awsRegion = package.devops.awsRegion; + +// Arguments checks + +const stage = process.argv[2] || 'dev'; +const sName = process.argv[3] || 'all'; + +const deployBase = function(basePath) { + const baseDirectory = basePath[0]; + const serverlessFile = baseDirectory + '/' + basePath[1]; + + let handlerFile; + if (basePath[2]) handlerFile = baseDirectory + '/' + basePath[2]; + else handlerFile = baseDirectory + '/' + 'handler.ts'; + + return new Listr( + [ + { + title: 'Remove handler file', + // eslint-disable-next-line @typescript-eslint/no-unused-vars + task: () => execa('rm', ['-f', 'handler.ts']).then((result) => {}), + }, + { + title: 'Copy handler file', + task: () => { + console.log('handler file here', handlerFile); + execa('cp', [handlerFile, 'handler.ts']).then((result) => {}); + }, + }, + { + title: 'Deploy serverless file', + task: async () => { + console.log('serverless file here', serverlessFile); + + const subprocess = execa( + 'npx', + ['sls', 'deploy', '--config', serverlessFile, '--stage', stage, '--aws-profile', awsProfile, '--region', awsRegion], + { env: { 'SLS_DEBUG=*': '*' } }, + ); + + console.log('deploying logs here', '.slslogs.txt'); + subprocess.stdout.pipe(fs.createWriteStream('.slslogs.txt', { flags: 'a' })); + + const { stdout } = await subprocess; + }, + }, + { + title: 'Remove handler file', + task: () => execa('rm', ['-f', 'handler.ts']).then((result) => {}), + }, + ], + { concurrent: false }, + ); +}; + +let functions = Object.keys(paths); + +if (sName != 'all') { + functions = sName.split(','); + for (let index = 0; index < functions.length; index++) { + if (!paths.hasOwnProperty(functions[index])) throw `Bad functions name for ${functions[index]}`; + } +} + +const tasksList = []; + +for (let index = 0; index < functions.length; index++) { + const element = functions[index]; + tasksList.push({ + title: 'Deploying' + ' ' + element, + task: () => { + return deployBase(paths[element]); + }, + }); +} + +const tasks = new Listr(tasksList, { concurrent: false }); + +tasks.run().catch((err) => { + console.error(err); +}); diff --git a/lib/seed/devops/deploy.js b/lib/seed/devops/deploy.js new file mode 100644 index 0000000..b230566 --- /dev/null +++ b/lib/seed/devops/deploy.js @@ -0,0 +1,103 @@ +/* eslint-disable @typescript-eslint/no-empty-function */ +/* eslint-disable @typescript-eslint/explicit-function-return-type */ +/* eslint-disable @typescript-eslint/no-var-requires */ +const execa = require('execa'); +const Listr = require('listr'); +const package = require('../../../package.json'); + +const fs = require('fs'); + +if (!package.devops) throw `No devops configuration found in package.json`; +if (!package.devops.services) throw `No devops services found in package.json`; +if (!package.devops.awsProfile) throw `No devops awsProfile found in package.json`; +if (!package.devops.awsRegion) throw 'No devops awsRegion found in package.json'; + +// Config input +const paths = package.devops.services; +const awsProfile = package.devops.awsProfile; +const awsRegion = package.devops.awsRegion; + +// Arguments checks + +const stage = process.argv[2] || 'dev'; +const sName = process.argv[3] || 'all'; + +const deployBase = function(basePath) { + const baseDirectory = basePath[0]; + const serverlessFile = baseDirectory + '/' + basePath[1]; + + let handlerFile; + if (basePath[2]) handlerFile = baseDirectory + '/' + basePath[2]; + else handlerFile = baseDirectory + '/' + 'handler.ts'; + + return new Listr( + [ + { + title: 'Remove handler file', + // eslint-disable-next-line @typescript-eslint/no-unused-vars + task: () => execa('rm', ['-f', 'handler.ts']).then((result) => {}), + }, + { + title: 'Copy handler file', + task: () => { + console.log('handler file here', handlerFile); + execa('cp', [handlerFile, 'handler.ts']).then((result) => {}); + }, + }, + { + title: 'Copy serverless file', + task: () => { + console.log('serverless file here', serverlessFile); + execa('cp', [serverlessFile, 'serverless.yaml']).then((result) => {}); + }, + }, + { + title: 'Deploy serverless file', + task: async () => { + const subprocess = execa( + 'npx', + ['sls', 'deploy', /*'--config', serverlessFile,*/ '--stage', stage, '--aws-profile', awsProfile, '--region', awsRegion], + { env: { 'SLS_DEBUG=*': '*' } }, + ); + + console.log('deploying logs here', '.slslogs.txt'); + subprocess.stdout.pipe(fs.createWriteStream('.slslogs.txt', { flags: 'a' })); + + const { stdout } = await subprocess; + }, + }, + { + title: 'Remove handler file', + task: () => execa('rm', ['-f', 'handler.ts']).then((result) => {}), + }, + ], + { concurrent: false }, + ); +}; + +let functions = Object.keys(paths); + +if (sName != 'all') { + functions = sName.split(','); + for (let index = 0; index < functions.length; index++) { + if (!paths.hasOwnProperty(functions[index])) throw `Bad functions name for ${functions[index]}`; + } +} + +const tasksList = []; + +for (let index = 0; index < functions.length; index++) { + const element = functions[index]; + tasksList.push({ + title: 'Deploying' + ' ' + element, + task: () => { + return deployBase(paths[element]); + }, + }); +} + +const tasks = new Listr(tasksList, { concurrent: false }); + +tasks.run().catch((err) => { + console.error(err); +}); diff --git a/lib/seed/devops/handler-admin.ts b/lib/seed/devops/handler-admin.ts new file mode 100644 index 0000000..b796ab6 --- /dev/null +++ b/lib/seed/devops/handler-admin.ts @@ -0,0 +1,10 @@ +import 'reflect-metadata'; +import { ServerInstance } from '@seed/graphql/Server'; + +import { AdminResolvers } from '@src/__indexes/__resolvers.index.admin'; + +export async function adminHandler(event: any, context: any): Promise { + const apolloInstance = await ServerInstance.getInstance(AdminResolvers); + const response = await apolloInstance.run(event, context); + return response; +} diff --git a/lib/seed/devops/handler-app.ts b/lib/seed/devops/handler-app.ts new file mode 100644 index 0000000..b1feef5 --- /dev/null +++ b/lib/seed/devops/handler-app.ts @@ -0,0 +1,9 @@ +import 'reflect-metadata'; +import { ServerInstance } from '@seed/graphql/Server'; +import { AppResolvers } from '@src/__indexes/__resolvers.index'; + +export async function appHandler(event: any, context: any): Promise { + const apolloInstance = await ServerInstance.getInstance(AppResolvers); + const response = await apolloInstance.run(event, context); + return response; +} diff --git a/lib/seed/devops/handler-hooks.ts b/lib/seed/devops/handler-hooks.ts new file mode 100644 index 0000000..4c174a8 --- /dev/null +++ b/lib/seed/devops/handler-hooks.ts @@ -0,0 +1,153 @@ +/* eslint-disable @typescript-eslint/no-use-before-define */ +/* eslint-disable @typescript-eslint/explicit-function-return-type */ +import 'reflect-metadata'; + +import { modelsLoaders } from '@src/__indexes/__loaders'; +import ChangeStreamModel from '@lib/seed/services/change-stream/change-stream.model'; +import { StreamOperationType, PostHookStatus } from '@lib/seed/services/change-stream/change-stream.components'; +import { AsyncHooksService } from '@lib/__hooks'; +import { createApolloContext } from '@seed/graphql/Middleware'; +import { SettingsCache } from '@seed/graphql/Settings'; + +export const hookHandler = async (event: any): Promise => { + const settingsI = SettingsCache.getInstance(); + await settingsI.refreshCache(); + + process.env.isHook = 'true'; + + const { streamId, notificationRessourceId } = event; + + if (streamId) { + await streamHandler(streamId); + } else if (notificationRessourceId) { + await notificationHandler(notificationRessourceId); + } else { + console.log('[ASYNC - ERROR]', 'was called with no info'); + } + + return; +}; + +async function notificationHandler(streamId: string) { + const stream = await (await new ChangeStreamModel().db()).findOneAndUpdate( + { _id: streamId, hookStatus: PostHookStatus.new }, + { $set: { hookStatus: PostHookStatus.inProcess } }, + ); + + if (stream && stream.value) { + const streamData = stream.value; + console.log('[ASYNC - STREAM] Stream', stream.value.collection, stream.value.operation); + + try { + if (streamData.operation == StreamOperationType.hook) { + const HookService = new AsyncHooksService(); + + if (!HookService[streamData.collection]) + throw `[ASYNC - STREAM] ERROR - No AsyncHooksService for this ${streamData.collection}. Are you sure you put it in the AsyncHooksService ?`; + + if (streamData.insertedValues && streamData.insertedValues.length > 0) { + const paramsArray = streamData.insertedValues as any[]; + + for (let index = 0; index < paramsArray.length; index++) { + // Deal with CTX + if (paramsArray[index].ctx) { + paramsArray[index] = await createApolloContext(paramsArray[index].ctx.user._id, paramsArray[index].ctx.organisationId); + } + } + await HookService[streamData.collection](...paramsArray); + } else await HookService[streamData.collection](); + } else { + if (!modelsLoaders[streamData.collection]) + throw `[ASYNC - STREAM] ERROR - No ModelLoaders for this ${streamData.collection}. Are you sure you put it in the ModelLoaders ?`; + + console.log('streamData', streamData); + const model = modelsLoaders[streamData.collection]; + const modelData = await model.getOne({ _id: streamData.documentKey }, null); + console.log('model', model); + + switch (streamData.operation) { + case StreamOperationType.insert: + await model.afterCreate(); + break; + case StreamOperationType.update: + await model.afterUpdate(); + break; + case StreamOperationType.delete: + await model.afterDelete(); + break; + default: + throw `[ASYNC - STREAM] ERROR - No operation ${streamData.operation} for this ${streamData.collection}`; + } + } + await (await new ChangeStreamModel().db()).updateOne({ _id: streamId }, { $set: { hookStatus: PostHookStatus.completed } }); + console.log('[ASYNC - STREAM] COMPLETE', { _id: streamId }); + } catch (error) { + console.error('[ASYNC - STREAM] ERROR', error); + await (await new ChangeStreamModel().db()).updateOne({ _id: streamId }, { $set: { hookStatus: PostHookStatus.error } }); + } + } else { + console.error('[ASYNC - STREAM] ERROR', 'No new stream found', streamId); + } +} + +async function streamHandler(streamId: string) { + const stream = await (await new ChangeStreamModel().db()).findOneAndUpdate( + { _id: streamId, hookStatus: PostHookStatus.new }, + { $set: { hookStatus: PostHookStatus.inProcess } }, + ); + + if (stream && stream.value) { + const streamData = stream.value; + console.log('[ASYNC - STREAM] Stream', stream.value.collection, stream.value.operation); + + try { + if (streamData.operation == StreamOperationType.hook) { + const HookService = new AsyncHooksService(); + + if (!HookService[streamData.collection]) + throw `[ASYNC - STREAM] ERROR - No AsyncHooksService for this ${streamData.collection}. Are you sure you put it in the AsyncHooksService ?`; + + if (streamData.insertedValues && streamData.insertedValues.length > 0) { + const paramsArray = streamData.insertedValues as any[]; + + for (let index = 0; index < paramsArray.length; index++) { + // Deal with CTX + if (paramsArray[index].ctx) { + paramsArray[index] = await createApolloContext(paramsArray[index].ctx.user._id, paramsArray[index].ctx.organisationId); + } + } + await HookService[streamData.collection](...paramsArray); + } else await HookService[streamData.collection](); + } else { + if (!modelsLoaders[streamData.collection]) + throw `[ASYNC - STREAM] ERROR - No ModelLoaders for this ${streamData.collection}. Are you sure you put it in the ModelLoaders ?`; + + console.log('streamData', streamData); + const model = modelsLoaders[streamData.collection]; + const modelData = await model.getOne({ _id: streamData.documentKey }, null); + console.log('model', model); + + switch (streamData.operation) { + case StreamOperationType.insert: + await model.afterCreate(); + break; + case StreamOperationType.update: + await model.afterUpdate(); + break; + case StreamOperationType.delete: + await model.afterDelete(); + break; + default: + throw `[ASYNC - STREAM] ERROR - No operation ${streamData.operation} for this ${streamData.collection}`; + } + } + await (await new ChangeStreamModel().db()).updateOne({ _id: streamId }, { $set: { hookStatus: PostHookStatus.completed } }); + console.error('[ASYNC - STREAM] COMPLETE', { _id: streamId }); + } catch (error) { + console.error('[ASYNC - STREAM] ERROR', error); + await (await new ChangeStreamModel().db()).updateOne({ _id: streamId }, { $set: { hookStatus: PostHookStatus.error } }); + } + } else { + console.error('[ASYNC - STREAM] ERROR', 'No new stream found', streamId); + } +} diff --git a/lib/seed/devops/serverless-admin.yaml b/lib/seed/devops/serverless-admin.yaml new file mode 100644 index 0000000..d9e652a --- /dev/null +++ b/lib/seed/devops/serverless-admin.yaml @@ -0,0 +1,55 @@ +app: ${file(./package.json):name} +service: admin +package: + individually: true +provider: + name: aws + stage: ${opt:stage,'dev'} + runtime: nodejs12.x + environment: + AWS_Hooks: hooks-${self:provider.stage}-hookHandler + S3_BUCKET: ${self:app}-appdata + iamRoleStatements: + - Effect: Allow + Action: + - lambda:InvokeFunction + Resource: '*' + - Effect: Allow + Action: + - s3:* + Resource: arn:aws:s3:::${self:provider.environment.S3_BUCKET}/* +plugins: + - serverless-webpack + - serverless-domain-manager +custom: + stage: ${opt:stage, self:provider.stage} + domains: ${file(./config/domains.yaml):domains} + customDomain: ${file(./config/domains.yaml):customDomain} + webpack: + webpackConfig: './lib/seed/webpack.config.js' # Name of webpack configuration file + includeModules: + forceExclude: + - aws-sdk + - puppeteer +functions: + admin: + handler: handler.adminHandler + timeout: 60 + events: + - http: + path: / + method: post + cors: + origin: '*' + headers: + - Content-Type + - Authorization + - X-Api-Key + - X-Amz-Security-Token + - X-Amz-User-Agent + - organisationid + + - http: + path: / + method: get + cors: true diff --git a/lib/seed/devops/serverless-app.yaml b/lib/seed/devops/serverless-app.yaml new file mode 100644 index 0000000..b15e7ab --- /dev/null +++ b/lib/seed/devops/serverless-app.yaml @@ -0,0 +1,60 @@ +app: ${file(./package.json):name} +service: app +package: + individually: true +provider: + name: aws + stage: ${opt:stage,'dev'} + runtime: nodejs12.x + environment: + AWS_Hooks: hooks-${self:provider.stage}-hookHandler + S3_BUCKET: ${self:app}-appdata + iamRoleStatements: + - Effect: Allow + Action: + - lambda:InvokeFunction + Resource: '*' + - Effect: Allow + Action: + - s3:* + Resource: arn:aws:s3:::${self:provider.environment.S3_BUCKET}/* + +plugins: + - serverless-webpack + - serverless-domain-manager + - serverless-offline +custom: + stage: ${opt:stage, self:provider.stage} + domains: ${file(./config/domains.yaml):domains} + customDomain: ${file(./config/domains.yaml):customDomain} + serverless-offline: + httpPort: 4000 + lambdaPort: 4010 + webpack: + webpackConfig: './lib/seed/webpack.config.js' # Name of webpack configuration file + includeModules: + forceExclude: + - aws-sdk + - puppeteer +functions: + app: + handler: handler.appHandler + timeout: 60 + events: + - http: + path: / + method: post + cors: + origin: '*' + headers: + - Content-Type + - Authorization + - X-Authorization + - X-Api-Key + - X-Amz-Security-Token + - X-Amz-User-Agent + - organisationid + - http: + path: / + method: get + cors: true diff --git a/lib/seed/devops/serverless-domains.yaml b/lib/seed/devops/serverless-domains.yaml new file mode 100644 index 0000000..9008ff8 --- /dev/null +++ b/lib/seed/devops/serverless-domains.yaml @@ -0,0 +1,12 @@ +app: ${file(./package.json):name} +service: domains +provider: + name: aws + stage: ${opt:stage,'dev'} + runtime: nodejs12.x +plugins: + - serverless-domain-manager +custom: + stage: ${opt:stage, self:provider.stage} + domains: ${file(./config/domains.yaml):domains} + customDomain: ${file(./config/domains.yaml):customDomain} \ No newline at end of file diff --git a/lib/seed/devops/serverless-hooks.yaml b/lib/seed/devops/serverless-hooks.yaml new file mode 100644 index 0000000..044d425 --- /dev/null +++ b/lib/seed/devops/serverless-hooks.yaml @@ -0,0 +1,37 @@ +app: ${file(./package.json):name} +service: hooks +package: + individually: true +provider: + name: aws + stage: ${opt:stage,'dev'} + runtime: nodejs12.x + environment: + AWS_Hooks: hooks-${self:provider.stage}-hookHandler + S3_BUCKET: ${self:app}-appdata + iamRoleStatements: + - Effect: Allow + Action: + - lambda:InvokeFunction + Resource: '*' + - Effect: Allow + Action: + - s3:* + Resource: arn:aws:s3:::${self:provider.environment.S3_BUCKET}/* + +plugins: + - serverless-webpack +custom: + stage: ${opt:stage, self:provider.stage} + webpack: + webpackConfig: './lib/seed/webpack.config.js' # Name of webpack configuration file + includeModules: + forceExclude: + - aws-sdk + - puppeteer +functions: + hookHandler: + handler: handler.hookHandler + memorySize: 1024 # optional, in MB, default is 1024 + timeout: 600 # optional, in seconds, default is 6 + async: true diff --git a/lib/seed/engine/EngineAccessService.ts b/lib/seed/engine/EngineAccessService.ts new file mode 100644 index 0000000..b1e6652 --- /dev/null +++ b/lib/seed/engine/EngineAccessService.ts @@ -0,0 +1,161 @@ +import { AccountTypeEnum } from '@src/accounts/account.components'; +import AccountModel from '@src/accounts/account.model'; +import { newError } from '@seed/helpers/Error'; +import _ from 'lodash'; + +export class EngineAccessService { + static addPermissions(ressource: any, type: ('r' | 'w' | 'd')[], ids: (string | string)[]): void { + for (let index = 0; index < type.length; index++) { + const t = type[index]; + ressource.permissions[t] = ressource.permissions[t].concat(ids); + } + } + + static addPermissionToQuery(account: AccountModel | null, query: 'get' | 'update' | 'delete', params: any): any { + let types: any[] = ['public']; + + if (account && account.types) { + /* + * Verify if admin, no need to add the query filters + */ + if (account.types.includes(AccountTypeEnum.admin)) return params; + + /* + * Verify if organisation type of access + */ + + // if (account.organisationIds) { + // params.organisationId = { $in: account.organisationIds }; + // } + + /* + * Add the account id and type + */ + + types.push(account._id); + types = types.concat(account.types); + } + + if (!params.$and) params.$and = []; + + switch (query) { + default: + case 'get': + params.$and.push({ 'permissions.r': { $in: types } }); + break; + case 'update': + params.$and.push({ 'permissions.w': { $in: types } }); + break; + case 'delete': + params.$and.push({ 'permissions.d': { $in: types } }); + break; + } + + return params; + } + + static checkPermissions(ressource: any, account: AccountModel | null, type: 'c' | 'r' | 'w' | 'd'): boolean { + // Adding the public by default + let perm: any[] = [AccountTypeEnum.public]; + + if (account && account.types) { + /* + * Verify if admin + */ + if (account.types.includes(AccountTypeEnum.admin)) return true; + + /* + * Verify if organisation type of access + */ + if ( + ressource.organisationId && + account.organisationIds && + !account.organisationIds.includes(ressource.organisationId) && + !ressource.permissions.r.includes(AccountTypeEnum.public) + ) + throw newError(2100, { + allowed: ressource.organisationId, + you: account.organisationIds, + }); + + /* + * Add the account id and type + */ + perm.push(account._id); + perm = perm.concat(account.types); + } + + /* + * Verify on the ressource level + */ + + const permissions = ressource.permissions[type]; + + // if there is no permission on the ressource, return true + if (!permissions) return true; + + let hasPerm = false; + + // Verifying if it matches + for (let index = 0; index < perm.length; index++) { + const element = perm[index]; + if (permissions.includes(element)) { + hasPerm = true; + break; + } + } + + if (!hasPerm) + throw newError(2000, { + allowed: permissions, + you: { _id: account?._id, types: account?.types }, + }); + + return true; + } + + static checkOrganisationPermissions(ressource: any, organisationId: string): boolean { + if (organisationId == ressource.organisationId || organisationId == ressource._id) return true; + throw newError(2100, { allowedOrgId: ressource.organisationId, youOrgId: organisationId }); + } + + static addOrganisationToQuery(account: AccountModel | null, query: 'get' | 'update' | 'delete', params: any): any { + let types: any[] = ['public']; + + if (account && account.types) { + /* + * Verify if admin, no need to add the query filters + */ + if (account.types.includes(AccountTypeEnum.admin)) return params; + + /* + * Verify if organisation type of access + */ + if (account.organisationIds) { + params.organisationId = { $in: account.organisationIds }; + } + + /* + * Add the account id and type + */ + + types.push(account._id); + types = types.concat(account.types); + } + + switch (query) { + default: + case 'get': + params = { ...params, $or: [{ 'permissions.r': { $in: _.uniq(types) } }] }; + break; + case 'update': + params = { ...params, $or: [{ 'permissions.w': { $in: _.uniq(types) } }] }; + break; + case 'delete': + params = { ...params, $or: [{ 'permissions.d': { $in: _.uniq(types) } }] }; + break; + } + + return params; + } +} diff --git a/lib/seed/engine/EngineModel.ts b/lib/seed/engine/EngineModel.ts new file mode 100644 index 0000000..748256f --- /dev/null +++ b/lib/seed/engine/EngineModel.ts @@ -0,0 +1,277 @@ +import _ from 'lodash'; + +import { ObjectType } from 'type-graphql'; +import { classToPlain, Exclude, plainToClassFromExist } from 'class-transformer'; + +import { Collection, ObjectId } from 'mongodb'; +import DB from '@seed/services/database/DBService'; + +import { Permission } from '@seed/interfaces/permission'; +import { ApolloContext } from '@seed/interfaces/context'; + +import { IEngineSchema, MetaBy } from './EngineSchema'; + +import { GetArgs } from '@seed/graphql/Request'; + +import { + bulk, + deleteOne, + getAll, + getCount, + getMany, + getOne, + getQuery, + saveMany, + saveOne, + updateOne, + updateOneCustom, + updateOneSubField, +} from './utils/crud.utils'; +import { getCountGeneric, getManyGeneric, getManyGenericWithArgs, getOneGeneric } from './utils/service.utils'; + +import { + BulkInput, + DeleteOneInput, + GetAllInput, + GetCountInput, + GetManyInput, + GetOneInput, + GetQueryInput, + SaveManyInput, + SaveOneInput, + UpdateOneCustomInput, + UpdateOneInput, + UpdateOneSubField, +} from './utils/__interface'; +import { EnginePathComponent } from '@seed/interfaces/components'; +import { ModelCollectionEnum } from '@src/__indexes/__collections'; +import { nullToUndefined } from '@seed/helpers/Utils'; + +@ObjectType() +export abstract class EngineModel implements IEngineSchema { + @Exclude() + public collectionName: ModelCollectionEnum | string; + @Exclude() + public permissions: Permission; + + @Exclude() + defaultSort: string; + + dbData: SchemaDB; + + searchT?: string; + + _id: string; + organisationId?: string | undefined; + paths?: EnginePathComponent[]; + by?: MetaBy | undefined; + createdAt: Date; + updatedAt: Date; + + tagsIds?: string[]; + + public constructor(init: { + collectionName: ModelCollectionEnum | string; + permissions: Permission; + dataInit: SchemaDB & Partial; + defaultSort?: string; + }) { + this.collectionName = init.collectionName; + this.permissions = init.permissions; + + this.dbData = init.dataInit; + + this.permissions = { + c: init.permissions.c, + r: init.permissions.r, + w: init.permissions.w, + d: init.permissions.d, + }; + + this.createdAt = new Date(); + this.updatedAt = new Date(); + + this.defaultSort = init.defaultSort || 'createdAt desc'; + } + + /* + * + * Model functions + * + */ + + public get(): Schema { + const data: any = this.dbData; + + if (this._id) data._id = this._id; + if (this.permissions) data.permissions = this.permissions; + + if (this.organisationId) data.organisationId = this.organisationId; + if (this.by) data.by = this.by; + + if (this.createdAt) data.createdAt = this.createdAt; + if (this.updatedAt) data.updatedAt = this.updatedAt; + + return this.plainToClass(data) as Schema; + } + + public set(doc: any): void { + const newData = { + ...classToPlain(this.dbData), // Get old data + ...doc, // Replace with new data + }; + + this.dbData = _.assign(this.dbData, newData); + + if (doc._id) this._id = doc._id; + + if (doc.permissions) { + this.permissions = { + c: _.uniq(_.concat(doc.permissions.c)), + r: _.uniq(_.concat(doc.permissions.r)), + w: _.uniq(_.concat(doc.permissions.w)), + d: _.uniq(_.concat(doc.permissions.d)), + }; + } + + if (doc.organisationId) this.organisationId = doc.organisationId; + + if (doc.paths) this.paths = doc.paths; + + if (doc.createdAt) this.createdAt = doc.createdAt; + if (doc.updatedAt) this.updatedAt = doc.updatedAt; + } + + async beforeCreate?(ctx?: ApolloContext | null): Promise; + async beforeUpdate?(ctx?: ApolloContext | null): Promise; + async beforeDelete?(ctx?: ApolloContext | null): Promise; + + async afterCreate?(changeStream: any): Promise; + async afterUpdate?(changeStream: any): Promise; + async afterDelete?(changeStream: any): Promise; + + /* + * + * Search Functions & Helpers + * + */ + + searchEngine?(): any; + searchText?(): any; + /* + * + * Transform Functions + * + */ + + abstract plainToClass(plain: any): Schema | Schema[]; + + /* + * + * DB Functions + * + */ + + public async db(): Promise> { + return await (await DB.getInstance()).db.collection(this.collectionName); + } + + generatePaths?(): EnginePathComponent[]; + + public getPaths(): EnginePathComponent[] { + return this.generatePaths ? this.generatePaths() : this.paths ? this.paths : []; + } + + public getFullPath(): EnginePathComponent[] { + const myPath = this.getPaths(); + myPath.push({ + ressourceModel: this.collectionName, + ressourceId: this._id, + }); + + return myPath; + } + + // ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + // ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + // ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + // ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + // ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + // ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + + public async getQuery(input: GetQueryInput): Promise { + return getQuery(this, input); + } + public async getMany(input: GetManyInput): Promise { + return getMany(this, input); + } + public async getCount(input: GetCountInput): Promise { + return getCount(this, input); + } + public async getAll(input: GetAllInput): Promise { + return getAll(this, input); + } + public async getOne(input: GetOneInput): Promise { + let newInput; + + if (!input.query) { + newInput = { query: input }; + } else newInput = input; + + return getOne(this, newInput); + } + + // ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗██╗ ██████╗ ███╗ ██╗ + // ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██║██╔═══██╗████╗ ██║ + // ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║██║ ██║██╔██╗ ██║ + // ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║██║ ██║██║╚██╗██║ + // ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ██║╚██████╔╝██║ ╚████║ + // ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═══╝ + + public async saveOne(input: SaveOneInput): Promise { + return saveOne(this, input); + } + public async updateOne(input: UpdateOneInput): Promise { + return updateOne(this, input); + } + public async updateOneCustom(input: UpdateOneCustomInput): Promise { + return updateOneCustom(this, input); + } + public async updateOneSubField(input: UpdateOneSubField): Promise { + return updateOneSubField(this, input); + } + + public async deleteOne(input: DeleteOneInput): Promise { + return deleteOne(this, input); + } + + public async saveMany(input: SaveManyInput): Promise { + return saveMany(this, input); + } + + public async bulk(input: BulkInput): Promise { + return bulk(this, input); + } + + // ██████╗ ███████╗███╗ ██╗███████╗██████╗ ██╗ ██████╗███████╗ + // ██╔════╝ ██╔════╝████╗ ██║██╔════╝██╔══██╗██║██╔════╝██╔════╝ + // ██║ ███╗█████╗ ██╔██╗ ██║█████╗ ██████╔╝██║██║ ███████╗ + // ██║ ██║██╔══╝ ██║╚██╗██║██╔══╝ ██╔══██╗██║██║ ╚════██║ + // ╚██████╔╝███████╗██║ ╚████║███████╗██║ ██║██║╚██████╗███████║ + // ╚═════╝ ╚══════╝╚═╝ ╚═══╝╚══════╝╚═╝ ╚═╝╚═╝ ╚═════╝╚══════ + + public async getOneGeneric(id: string, ctx: ApolloContext): Promise { + return getOneGeneric(this, id, ctx); + } + public async getManyGeneric(query: any, pagination: GetArgs | undefined, ctx: ApolloContext | null): Promise { + return getManyGeneric(this, query, pagination, ctx); + } + + public async getCountGeneric(baseArguments: any, ctx: ApolloContext): Promise { + return getCountGeneric(this, baseArguments, ctx); + } + + public async getManyGenericWithArgs(baseArguments: any, ctx: ApolloContext | null): Promise { + return getManyGenericWithArgs(this, baseArguments, ctx); + } +} diff --git a/lib/seed/engine/EngineSchema.ts b/lib/seed/engine/EngineSchema.ts new file mode 100644 index 0000000..8e6c159 --- /dev/null +++ b/lib/seed/engine/EngineSchema.ts @@ -0,0 +1,58 @@ +import { EnginePathComponent } from '@seed/interfaces/components'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { Field, ID, ObjectType, Authorized, InterfaceType } from 'type-graphql'; + +@ObjectType() +export class MetaBy { + @Field(() => ID, { nullable: true }) + createdBy?: string; + + @Field(() => ID, { nullable: true }) + updatedBy?: string; + + @Field(() => ID, { nullable: true }) + deletedBy?: string; +} + +@ObjectType() +export class MetaPermissions { + @Field(() => [String]) + r: string[]; + + @Authorized(AccountTypeEnum.admin) + @Field(() => [String]) + w: string[]; + + @Authorized(AccountTypeEnum.admin) + @Field(() => [String]) + d: string[]; +} + +@InterfaceType() +export abstract class IEngineSchema { + @Field(() => ID) + _id: string; + + @Field(() => ID, { nullable: true }) + organisationId?: string; + + @Field(() => [EnginePathComponent], { nullable: true }) + paths?: EnginePathComponent[]; + + searchT?: string; + + @Field(() => MetaBy, { nullable: true }) + by?: MetaBy; + + @Field(() => MetaPermissions, { nullable: true }) + permissions: MetaPermissions; + + @Field() + createdAt: Date; + + @Field() + updatedAt: Date; + + @Field(() => [String], { nullable: true }) + tagsIds?: string[]; +} diff --git a/lib/seed/engine/decorators/db.guard.ts b/lib/seed/engine/decorators/db.guard.ts new file mode 100644 index 0000000..99233ec --- /dev/null +++ b/lib/seed/engine/decorators/db.guard.ts @@ -0,0 +1,40 @@ +import { registerDecorator, ValidationOptions, ValidatorConstraint, ValidatorConstraintInterface, ValidationArguments } from 'class-validator'; +import { ModelCollectionEnum } from '@src/__indexes/__collections'; +import DB from '@seed/services/database/DBService'; + +import { newError } from '@seed/helpers/Error'; + +@ValidatorConstraint({ async: true }) +export class IsRefExistConstraint implements ValidatorConstraintInterface { + // eslint-disable-next-line @typescript-eslint/explicit-function-return-type + async validate(ressourceId: any, args: ValidationArguments & any) { + // Validate only when in the DB object + if (args.targetName.includes('DB')) { + const obj = args.object as any; + // Check what kind of id check + if (obj.ressourceId && obj.ressourceModel) { + + const res = await (await DB.getInstance()).db.collection(ModelCollectionEnum[obj.ressourceModel]).findOne({ _id: obj.ressourceId }); + if (!res) throw newError(4040, obj); + } else { + } + } + + args.hasValidated = true; + + return true; + } +} + +export function IsRefExist(validationOptions?: ValidationOptions) { + // eslint-disable-next-line @typescript-eslint/explicit-function-return-type + return function(object: Record, propertyName: string) { + registerDecorator({ + target: object.constructor, + propertyName: propertyName, + options: validationOptions, + constraints: [], + validator: IsRefExistConstraint, + }); + }; +} diff --git a/lib/seed/engine/decorators/pre.hooks.ts b/lib/seed/engine/decorators/pre.hooks.ts new file mode 100644 index 0000000..e69de29 diff --git a/lib/seed/engine/genericResolvers/BaseEngineResolver.ts b/lib/seed/engine/genericResolvers/BaseEngineResolver.ts new file mode 100644 index 0000000..5ebe023 --- /dev/null +++ b/lib/seed/engine/genericResolvers/BaseEngineResolver.ts @@ -0,0 +1,114 @@ +import { Query, Arg, Resolver, Ctx, Args, Mutation, ClassType } from 'type-graphql'; +import { ApolloContext } from '@seed/interfaces/context'; +import { EngineMiddleware, EngineMiddlewareInput } from '@seed/graphql/MiddlewareV2'; +import { EngineModel } from '../EngineModel'; + +export const createEngineQueryResolver = , T extends ClassType, ARGS extends ClassType>(init: { + domain: string; + schemaName: U; + modelName: T; + argsType: ARGS; + engineMiddleware: EngineMiddlewareInput; +}): any => { + @Resolver({ isAbstract: true }) + abstract class BaseQueryResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ + + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + @Query(() => init.schemaName, { name: `${init.domain}GetOne` }) + @EngineMiddleware(init.engineMiddleware) + async getOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = new init.modelName() as EngineModel; + const result = await model.getOneGeneric(id, ctx); + + return result; + } + + @Query(() => [init.schemaName], { name: `${init.domain}GetMany` }) + @EngineMiddleware(init.engineMiddleware) + async getMany(@Args(() => init.argsType) args: ARGS, @Ctx() ctx: ApolloContext): Promise { + const model = new init.modelName() as EngineModel; + return model.getManyGenericWithArgs(args, ctx); + } + + @Query(() => Number, { name: `${init.domain}GetCount` }) + @EngineMiddleware(init.engineMiddleware) + async getCount(@Args(() => init.argsType) args: ARGS, @Ctx() ctx: ApolloContext): Promise { + const model = new init.modelName() as EngineModel; + return model.getCountGeneric(args, ctx); + } + } + + return BaseQueryResolver; +}; + +export const createEngineMutationResolver = < + U extends ClassType, + T extends ClassType, + NEW extends ClassType, + EDIT extends ClassType +>(init: { + domain: string; + schemaName: U; + modelName: T; + newInput: NEW; + editInput: EDIT; + engineMiddleware: EngineMiddlewareInput; + validate?: boolean; +}): any => { + @Resolver({ isAbstract: true }) + abstract class BaseMutationResolver { + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + @Mutation(() => init.schemaName, { name: `${init.domain}AddOne` }) + @EngineMiddleware({ ...init.engineMiddleware, validations: [{ schema: init.newInput }] }) + async addOne(@Arg('input', () => init.newInput) input: NEW, @Ctx() ctx: ApolloContext): Promise { + const model = new init.modelName() as EngineModel; + return await await model.saveOne({ newData: input, ctx }); + } + + @Mutation(() => init.schemaName, { name: `${init.domain}EditOne` }) + @EngineMiddleware({ ...init.engineMiddleware, validations: [{ schema: init.editInput }] }) + async editOne(@Arg('id') id: string, @Arg('input', () => init.editInput) input: EDIT, @Ctx() ctx: ApolloContext): Promise { + const model = new init.modelName() as EngineModel; + return await await model.updateOne({ + query: { _id: id }, + newData: input, + ctx: ctx, + }); + } + + @Mutation(() => init.schemaName, { name: `${init.domain}DeleteOne` }) + @EngineMiddleware({ ...init.engineMiddleware }) + async deleteOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = new init.modelName() as EngineModel; + return await await model.deleteOne({ + query: { _id: id }, + ctx: ctx, + }); + } + } + + return BaseMutationResolver; +}; diff --git a/lib/seed/engine/utils/__interface.ts b/lib/seed/engine/utils/__interface.ts new file mode 100644 index 0000000..bf6bfbe --- /dev/null +++ b/lib/seed/engine/utils/__interface.ts @@ -0,0 +1,76 @@ +import { GetArgs } from '@seed/graphql/Request'; +import { ApolloContext } from '@seed/interfaces/context'; +import { FilterQuery, UpdateQuery } from 'mongodb'; +import { EngineModel } from '../EngineModel'; +import { IEngineSchema } from '../EngineSchema'; + +export interface GetQueryInput { + query: FilterQuery; + ctx?: ApolloContext | null; +} + +export interface GetManyInput { + query: FilterQuery; + pagination?: GetArgs; + ctx?: ApolloContext | null; +} +export interface GetAllInput { + query: FilterQuery; + ctx?: ApolloContext | null; +} +export interface GetCountInput { + query: FilterQuery; + ctx?: ApolloContext | null; +} + +export interface GetOneInput { + query: FilterQuery; + ctx?: ApolloContext | null; +} + +export interface SaveOneInput { + newData?: SchemaDBInterface & Partial; + additionnalData?: Partial; + ctx?: ApolloContext | null; + upsert?: boolean; +} + +export interface SaveManyInput { + models: EngineModel[]; + ctx?: ApolloContext | null; +} + +export interface BulkInput { + inserts?: EngineModel[]; + updates?: { + query: FilterQuery; + newData?: Partial; + updateRequest?: UpdateQuery; + }[]; + ctx?: ApolloContext | null; +} + +export interface UpdateOneInput { + query: FilterQuery; + newData: Partial; + ctx?: ApolloContext | null; +} + +export interface UpdateOneCustomInput { + query: FilterQuery; + updateRequest: UpdateQuery; + ctx?: ApolloContext | null; +} + +export interface UpdateOneSubField { + query: FilterQuery; + fieldName: keyof SchemaDBInterface; + subPath?: string; + fieldValue: any; + ctx?: ApolloContext | null; +} + +export interface DeleteOneInput { + query: FilterQuery; + ctx?: ApolloContext | null; +} diff --git a/lib/seed/engine/utils/crud.utils.ts b/lib/seed/engine/utils/crud.utils.ts new file mode 100644 index 0000000..f9886c3 --- /dev/null +++ b/lib/seed/engine/utils/crud.utils.ts @@ -0,0 +1,721 @@ +import { parsePaginationOptions } from '@lib/seed/helpers/Request'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { validateOrReject } from 'class-validator'; +import { EngineAccessService } from '../EngineAccessService'; +import { EngineModel } from '../EngineModel'; +import { IEngineSchema } from '../EngineSchema'; + +import { v4 as uuid } from 'uuid'; +import { + BulkInput, + DeleteOneInput, + GetAllInput, + GetCountInput, + GetManyInput, + GetOneInput, + GetQueryInput, + SaveManyInput, + SaveOneInput, + UpdateOneCustomInput, + UpdateOneInput, + UpdateOneSubField, +} from './__interface'; +import DB from '@lib/seed/services/database/DBService'; +import { createChangeStream } from './streams/stream.service'; +import { StreamOperationType } from './streams/schemas/stream.components'; +import { BulkWriteOperation, ObjectId } from 'mongodb'; +import { newError } from '@seed/helpers/Error'; +import { clog } from '@seed/helpers/Utils'; + +export function getQuery( + model: EngineModel, + input: GetQueryInput, +): any { + let finalQuery: any; + const { query, ctx } = input; + + if (ctx) { + const account = ctx.ctx.user; + const organisationId = ctx.ctx.organisationId; + + // We add the permission (on r field) to make sure that kind of user has access to the ressource + if (!ctx.ctx.noPermissionCheck) finalQuery = EngineAccessService.addPermissionToQuery(account, 'get', { ...query }); + + // We check if it comes from an organisation + // Check of the user's correct authorisation on that organisation comes from the middleware + if (!ctx.ctx.noOrganisationCheck) { + if (organisationId) { + if (model.collectionName == 'accounts') finalQuery['organisationIds'] = organisationId; + else finalQuery['organisationId'] = { $eq: organisationId }; + } else { + // If admin & public query -> do not add model condition and let search every records) + if (account && account.types && !account.types.includes(AccountTypeEnum.public) && !account.types.includes(AccountTypeEnum.admin)) { + finalQuery['organisationId'] = { $exists: false }; + } + } + } + } else { + finalQuery = { ...query }; + } + + return finalQuery; +} + +export const getMany = async ( + model: EngineModel, + input: GetManyInput, +): Promise => { + let { pagination } = input; + + const finalQuery = getQuery(model, input); + + if (!pagination) pagination = { limit: 100, skip: 0 }; + if (!pagination.sort) pagination.sort = model.defaultSort; + const paginationOption = parsePaginationOptions(pagination); + + try { + console.log('search request', JSON.stringify(finalQuery, null, 2)); + const dbResults = await (await model.db()).find(finalQuery, paginationOption).toArray(); + return model.plainToClass(dbResults) as any; + } catch (error) { + return []; + } +}; + +export const getCount = async ( + model: EngineModel, + input: GetCountInput, +): Promise => { + const finalQuery = getQuery(model, input); + + try { + const result = (await (await model.db()).countDocuments(finalQuery)) as any; + return result; + } catch (error) { + return 0; + } +}; + +export const getAll = async ( + model: EngineModel, + input: GetAllInput, +): Promise => { + const finalQuery = getQuery(model, input); + try { + clog(finalQuery); + const res = await (await model.db()).find(finalQuery).toArray(); + return model.plainToClass(res); + } catch (error) { + return []; + } +}; + +export const getOne = async ( + model: EngineModel, + input: GetOneInput, +): Promise => { + const { query, ctx } = input; + + let dataToReturn; + if (query._id) { + const _id = query._id as any; + if (_id.match(/^[0-9a-fA-F]{24}$/)) { + query._id = new ObjectId(_id) as any; + } + } + if (ctx) { + const account = ctx.ctx.user; + const organisationId = ctx.ctx.organisationId; + + const doc = await (await model.db()).findOne(query); + if (!doc) throw await newError(4041, { id: query._id, model: model.collectionName }); + + dataToReturn = model.plainToClass(doc) as any; + model.set(dataToReturn); + + if (!ctx.ctx.noOrganisationCheck) { + if (organisationId) { + if (model.collectionName != 'accounts') EngineAccessService.checkOrganisationPermissions(model, organisationId); + } + } + if (!ctx.ctx.noPermissionCheck) EngineAccessService.checkPermissions(dataToReturn, account, 'r'); + } else { + const doc = await (await model.db()).findOne(query); + if (!doc) throw await newError(4041, { id: query._id, model: model.collectionName }); + + dataToReturn = model.plainToClass(doc) as any; + model.set(dataToReturn); + } + + return model.get(); +}; + +export const saveOne = async ( + model: EngineModel, + input: SaveOneInput, +): Promise => { + const { newData, additionnalData, upsert } = input; + let ctx = input.ctx; + + if (newData) model.set(newData); + if (additionnalData) model.set(additionnalData); + + // Set the path + model.set({ paths: model.getPaths() }); + + if (ctx) { + const account = ctx.ctx.user; + const organisationId = ctx.ctx.organisationId; + + EngineAccessService.addPermissions(model, ['r', 'w', 'd'], [account._id]); + + // TODO : Check create permissions + + if (organisationId) { + model.set({ organisationId }); + } + + if (model.by) model.by.createdBy = account._id; + else { + model.by = { + createdBy: account._id, + }; + } + } + + // if (!ctx) ctx = await createApolloContext(); + + try { + if (model.beforeCreate) await model.beforeCreate(ctx); + await validateOrReject(model.dbData as any); + } catch (error) { + throw error; + } + + const dataToSave = model.get() as any; + if (model.searchText) dataToSave.searchT = model.searchText(); + + if (!model._id) model._id = uuid(); + + delete dataToSave._id; + + try { + const result = await (await model.db()).findOneAndUpdate( + { _id: model._id } as any, + { + $set: dataToSave, + $setOnInsert: { + _id: model._id, + }, + } as any, + { upsert: upsert ? upsert : true, returnOriginal: false }, + ); + + if (result.ok == 0) throw await newError(1000, result); + + await createChangeStream(model, StreamOperationType.insert); + + return model.get(); + } catch (error) { + throw await newError(1000, error); + } +}; + +export const updateOne = async ( + model: EngineModel, + input: UpdateOneInput, +): Promise => { + try { + const { query, newData, ctx } = input; + + if (!model._id) await model.getOne(input); + if (newData) model.set(newData); + + model.set({ paths: model.getPaths() }); + + if (ctx) { + const account = ctx.ctx.user; + // Check if doc exists & correct permissions + + if (model.by) model.by.updatedBy = account._id; + else { + model.by = { + updatedBy: account._id, + }; + } + + EngineAccessService.checkPermissions(model, account, 'w'); + } + + if (model.beforeUpdate) await model.beforeUpdate(ctx); + + const dataToSave = model.get() as any; + if (model.searchText) dataToSave.searchT = model.searchText(); + + const result = await (await model.db()).findOneAndUpdate(query, { $set: dataToSave }, { upsert: false, returnOriginal: false }); + if (result.ok != 1) throw await newError(1000, result); + + model.set(result.value); + + await createChangeStream(model, StreamOperationType.update); + + return model.get(); + } catch (error) { + throw error; + } +}; + +export const updateOneCustom = async ( + model: EngineModel, + input: UpdateOneCustomInput, +): Promise => { + try { + const { query, updateRequest, ctx } = input; + if (!model._id) await model.getOne(input); + + if (ctx) { + const account = ctx.ctx.user; + // Check if doc exists + // dataBeforeChange = model.get(); + + EngineAccessService.checkPermissions(model, account, 'w'); + + if (model.by) model.by.updatedBy = account._id; + else { + model.by = { + updatedBy: account._id, + }; + } + + // Validate new data that will go into the DB + // const validation = await model.validate(); + // if (!validation.success) return validation; + } + + // Prepare data + model.updatedAt = new Date(); + + const result = await (await model.db()).findOneAndUpdate(query, updateRequest, { upsert: false, returnOriginal: false }); + if (result.ok != 1) throw await newError(1000, result); + + if (result.value) model.set(result.value); + + await createChangeStream(model, StreamOperationType.update); + + return model.get(); + } catch (error) { + throw error; + } +}; + +export const deleteOne = async ( + model: EngineModel, + input: DeleteOneInput, +): Promise => { + const { query, ctx } = input; + try { + if (!model._id) await model.getOne(input); + if (model.beforeDelete) await model.beforeDelete(ctx); + + if (ctx) { + const account = ctx.ctx.user; + // Check if doc exists & correct permissions + EngineAccessService.checkPermissions(model, account, 'd'); + + if (model.by) model.by.deletedBy = account._id; + else { + model.by = { + deletedBy: account._id, + }; + } + } + + const result = await (await model.db()).findOneAndDelete(query); + if (result.ok != 1) throw await newError(1001); + + const deletedModel = '_deleted_' + model.collectionName; + await (await DB.getInstance()).db.collection(deletedModel).insertOne(model.get()); + + await createChangeStream(model, StreamOperationType.delete); + + return model.get(); + } catch (error) { + throw error; + } +}; + +export const saveMany = async ( + model: EngineModel, + input: SaveManyInput, +): Promise => { + const ctx = input.ctx; + const models = input.models; + + const dataToSaves: any[] = []; + for (let index = 0; index < models.length; index++) { + const element = models[index]; + + // Set the path + element.set({ paths: element.getPaths() }); + + if (ctx) { + const account = ctx.ctx.user; + const organisationId = ctx.ctx.organisationId; + + EngineAccessService.addPermissions(element, ['r', 'w', 'd'], [account._id]); + + // TODO : Check create permissions + + if (organisationId) { + element.set({ organisationId }); + } + if (element.by) element.by.createdBy = account._id; + else { + element.by = { + createdBy: account._id, + }; + } + try { + if (element.beforeCreate) await element.beforeCreate(ctx); + await validateOrReject(element.dbData); + } catch (error) { + throw error; + } + } + + const dataToSave = element.get() as any; + if (element.searchText) dataToSave.searchT = element.searchText(); + if (!element._id) dataToSave._id = uuid(); + + dataToSaves.push(dataToSave); + } + + try { + const result = await (await model.db()).insertMany(dataToSaves, { ordered: true }); + + if (result.result.ok == 0) throw await newError(1000, result); + + // await createChangeStream(model, StreamOperationType.insert); + + return dataToSaves; + } catch (error) { + throw await newError(1000, error); + } +}; + +export const bulk = async ( + model: EngineModel, + input: BulkInput, +): Promise => { + const { inserts, updates, ctx } = input; + + const bulkOps: BulkWriteOperation[] = []; + if (inserts) { + inserts.forEach((element) => { + bulkOps.push({ + insertOne: { + document: element.get() as any, + }, + }); + }); + } + + if (updates) { + updates.forEach((element) => { + if (element.newData) + bulkOps.push({ + updateOne: { + filter: element.query, + update: { $set: element.newData }, + }, + }); + else if (element.updateRequest) + bulkOps.push({ + updateOne: { + filter: element.query, + update: element.updateRequest, + }, + }); + }); + } + + const bOpsArray: BulkWriteOperation[][] = []; + if (bulkOps.length > 0) { + const chunk = 1000; + let i, j; + + for (i = 0, j = bulkOps.length; i < j; i += chunk) { + bOpsArray.push(bulkOps.slice(i, i + chunk)); + } + } else bOpsArray.push(bulkOps); + + try { + for (let index = 0; index < bOpsArray.length; index++) { + const element = bOpsArray[index]; + const res = await (await model.db()).bulkWrite(element); + + console.log(res); + } + } catch (error) { + throw await newError(1000, error); + } + + return; +}; + +export const updateOneSubField = async ( + model: EngineModel, + input: UpdateOneSubField, +): Promise => { + const { fieldName, fieldValue, subPath, query, ctx } = input; + + let updatePath = fieldName as string; + if (subPath) updatePath += subPath; + try { + const filterQ = { + ...query, + }; + return await model.updateOneCustom({ + query: filterQ, + updateRequest: { + $set: { + [`${updatePath}`]: fieldValue, + } as any, + }, + ctx, + }); + } catch (error) { + throw error; + } +}; + +// public async saveMany(newDataArr: Partial[], ctx: ApolloContext | null): Promise { +// // Initialize the change stream +// const changeStreams: any[] = []; +// const afterCreatePromises: any[] = []; + +// const prehooksPromises = newDataArr.map(async (newData) => { +// const clone = Object.assign(Object.create(Object.getPrototypeOf(this)), this); +// if (ctx) { +// const account = ctx.ctx.user; +// const organisationId = ctx.ctx.organisationId; + +// addPermissions(clone, ['r', 'w', 'd'], [account._id]); + +// if (organisationId) { +// newData.organisationId = organisationId; +// } + +// newData.createdBy = account._id; +// } + +// clone.set(newData); + +// if (!ctx) ctx = await createApolloContext(); + +// if (clone.prehook) await clone.prehook(); +// if (clone.beforeCreate) await clone.beforeCreate(ctx); + +// const { _id, ...dataToSave } = clone.get(); +// const savedId = _id ? _id : uuid(); +// clone.set({ _id: savedId }); + +// const changeStreamId = uuid(); +// const changeStreamData = { +// _id: changeStreamId, +// operation: StreamOperationType.insert, +// collection: clone.collectionName, +// documentKey: clone._id, +// insertedValues: clone.get(), +// hookStatus: PostHookStatus.new, +// createdAt: new Date(), +// updateAt: new Date(), +// }; + +// changeStreams.push(changeStreamData); + +// // eslint-disable-next-line @typescript-eslint/no-this-alias +// afterCreatePromises.push(async () => { +// if (clone.afterCreate) { +// const res = await asyncHook(changeStreamData); + +// if (res) { +// await clone.afterCreate(changeStreamData); + +// await (await DB.getInstance()).db +// .collection('stream.changes') +// .updateOne({ _id: changeStreamId }, { $set: { hookStatus: PostHookStatus.completed } }); +// } +// } +// }); + +// return clone.get(); +// }); + +// try { +// const dataArr = await Promise.all(prehooksPromises); +// await (await this.db()).insertMany(dataArr, { ordered: false }); +// await (await DB.getInstance()).db.collection('stream.changes').insertMany(changeStreams, { ordered: false }); +// await promiseAll(afterCreatePromises); +// return dataArr; +// } catch (error) { +// console.error(error); +// throw await newError('db.error', error); +// } +// } + +// ██████╗ ███████╗██████╗ ███╗ ███╗██╗███████╗███████╗██╗ ██████╗ ███╗ ██╗███████╗ +// ██╔══██╗██╔════╝██╔══██╗████╗ ████║██║██╔════╝██╔════╝██║██╔═══██╗████╗ ██║██╔════╝ +// ██████╔╝█████╗ ██████╔╝██╔████╔██║██║███████╗███████╗██║██║ ██║██╔██╗ ██║███████╗ +// ██╔═══╝ ██╔══╝ ██╔══██╗██║╚██╔╝██║██║╚════██║╚════██║██║██║ ██║██║╚██╗██║╚════██║ +// ██║ ███████╗██║ ██║██║ ╚═╝ ██║██║███████║███████║██║╚██████╔╝██║ ╚████║███████║ +// ╚═╝ ╚══════╝╚═╝ ╚═╝╚═╝ ╚═╝╚═╝╚══════╝╚══════╝╚═╝ ╚═════╝ ╚═╝ ╚═══╝╚══════╝ + +// public async updateQueryWithPermission(query: FilterQuery, ctx: ApolloContext): Promise> { +// const account = ctx.ctx.user; +// const organisationId = ctx.ctx.organisationId; + +// // We add the permission (on r field) to make sure that kind of user has access to the ressource +// const finalQuery = addPermissionToQuery(account, 'get', { ...query }); + +// // We check if it comes from an organisation +// // Check of the user's correct authorisation on that organisation comes from the middleware +// if (organisationId) { +// finalQuery['organisationId'] = { $eq: organisationId }; +// } else { +// finalQuery['organisationId'] = { $exists: false }; +// } + +// return finalQuery; +// } + +// ███████╗██╗ ██╗██████╗ ██████╗ ███████╗███████╗███████╗ +// ██╔════╝██║ ██║██╔══██╗ ██╔══██╗██╔════╝██╔════╝██╔════╝ +// ███████╗██║ ██║██████╔╝█████╗██████╔╝█████╗ ███████╗███████╗ +// ╚════██║██║ ██║██╔══██╗╚════╝██╔══██╗██╔══╝ ╚════██║╚════██║ +// ███████║╚██████╔╝██████╔╝ ██║ ██║███████╗███████║███████║ +// ╚══════╝ ╚═════╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝╚══════╝╚══════╝ + +// public async addOneSubModel( +// parentQ: { _id: string }, +// subModelRefField: string, +// subModelFieldName: keyof this, +// subModel: SUB, +// subModelInput: any, +// ctx: ApolloContext | null, +// ): Promise { +// try { +// subModelInput[subModelRefField] = parentQ._id; +// const subM = await subModel.saveOne(subModelInput, ctx); +// await this.updateOneCustom(parentQ as any, { $push: { [`${subModelFieldName}`]: subM._id } } as any, null); +// return subM; +// } catch (error) { +// throw error; +// } +// } + +// public async editOneSubModel( +// subModel: SUB, +// subModelId: string, +// subModelInput: any, +// ctx: ApolloContext | null, +// ): Promise { +// try { +// // No need to check if belongs to the correct organisation, the middleware does that +// const subM = await subModel.updateOne({ _id: subModelId } as any, subModelInput, ctx); +// return subM; +// } catch (error) { +// throw error; +// } +// } + +// public async deleteOneSubModel( +// parentQ: FilterQuery, +// subModelFieldName: keyof this, +// subModel: SUB, +// subModelId: string, +// ctx: ApolloContext | null, +// ): Promise { +// try { +// const subM = await subModel.deleteOne({ _id: subModelId } as any, ctx); +// await this.updateOneCustom(parentQ, { $pull: { [`${subModelFieldName}`]: subM._id } } as any, null); +// return subM; +// } catch (error) { +// throw error; +// } +// } + +// public getOneSubRessource(subRessoureFieldName: keyof this | string, subRessoureId: string): any | undefined { +// try { +// const subress = _.get(this, subRessoureFieldName); +// if (subress) { +// const ind = _.findIndex(subress, { _id: subRessoureId }); +// if (ind !== -1) return subress[ind]; +// } +// return; +// } catch (error) { +// throw error; +// } +// } + +// public getDefaultSubRessource(subRessoureFieldName: keyof this | string): any | undefined { +// try { +// const subress = _.get(this, subRessoureFieldName); +// if (subress) { +// const ind = _.findIndex(subress, { default: true }); +// if (ind !== -1) return subress[ind]; +// return subress[0]; +// } +// return; +// } catch (error) { +// throw error; +// } +// } + +// public async addOneSubRessource( +// parentQ: { _id: string }, +// subRessoureFieldName: keyof this | string, +// subRessoure: any, +// ctx: ApolloContext | null, +// ): Promise { +// try { +// if (subRessoure.default && _.get(this, subRessoureFieldName)) { +// // Remove all default from others +// await this.updateOneCustom( +// { _id: this._id } as any, +// { +// $set: { [`${subRessoureFieldName}.$[].default`]: false } as any, +// }, +// ctx, +// ); +// } else subRessoure.default = false; + +// // Generate ID if not in the model +// if (!subRessoure._id) subRessoure._id = uuid(); +// await this.updateOneCustom(parentQ as any, { $push: { [`${subRessoureFieldName}`]: subRessoure } } as any, ctx); +// return subRessoure; +// } catch (error) { +// throw error; +// } +// } + +// public async deleteOneSubRessource( +// parentQ: { _id: string }, +// subRessoureFieldName: keyof this | string, +// subRessoureId: string, +// ctx: ApolloContext | null, +// ): Promise { +// try { +// if (!this.getOneSubRessource(subRessoureFieldName, subRessoureId)) throw await newError('notFound', '400'); + +// const filterQ = { +// ...parentQ, +// [`${subRessoureFieldName}._id`]: subRessoureId, +// }; + +// await this.updateOneCustom(filterQ as any, { $pull: { [`${subRessoureFieldName}`]: { _id: subRessoureId } } } as any, ctx); +// return subRessoureId; +// } catch (error) { +// throw error; +// } +// } diff --git a/lib/seed/engine/utils/service.utils.ts b/lib/seed/engine/utils/service.utils.ts new file mode 100644 index 0000000..14522e0 --- /dev/null +++ b/lib/seed/engine/utils/service.utils.ts @@ -0,0 +1,118 @@ +import { GetArgs } from '@lib/seed/graphql/Request'; +import { ApolloContext } from '@lib/seed/interfaces/context'; + +import { baseSearchFunction } from '@lib/seed/services/database/DBRequestService'; +import { EngineModel } from '../EngineModel'; + +/* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ +*/ + +export const getOneGeneric = async (model: any, id: string, ctx: ApolloContext): Promise => { + try { + return await (model as EngineModel).getOne({ query: { _id: id }, ctx }); + } catch (error) { + throw error; + } +}; +export const getManyGeneric = async (model: any, query: any, pagination: GetArgs | undefined, ctx: ApolloContext | null): Promise => { + return baseSearchFunction({ + model: model, + query: query, + pagination: pagination, + engine: true, + ctx: ctx, + }); +}; + +export const getCountGeneric = async (model: any, baseArguments: any, ctx: ApolloContext): Promise => { + // eslint-disable-next-line @typescript-eslint/no-unused-vars + const { pagination, ...args } = baseArguments; + + const result = await baseSearchFunction({ + model: model, + query: args, + count: true, + engine: true, + ctx: ctx, + }); + + return result; +}; + +export const getManyGenericWithArgs = async (model: any, baseArguments: any, ctx: ApolloContext | null): Promise => { + try { + const { pagination, ...args } = baseArguments; + return baseSearchFunction({ + model: model, + query: args, + pagination: pagination, + engine: true, + ctx: ctx, + }); + } catch (error) { + throw error; + } +}; + +// export const editOneGeneric = async (model: T, id: string, body: Partial, ctx: ApolloContext): Promise => { +// try { +// await model as EngineModel).updateOne({ _id: id } as any, body, ctx); +// return model as EngineModel).get(); +// } catch (error) { +// throw error; +// } +// }; + +// export const deleteOneGeneric = async (model: T, id: string, ctx: ApolloContext): Promise => { +// try { +// await model as EngineModel).deleteOne({ _id: id } as any, ctx); +// return model as EngineModel).get(); +// } catch (error) { +// throw error; +// } +// }; + +// export const permissionsAddOne = async (model: any, id: string, input: NewPermissionInput, ctx: ApolloContext): Promise => { +// try { +// await model as EngineModel).updateOneCustom({ _id: id }, { $addToSet: { [`${input.permissionType}`]: input.permission } }, ctx); +// return model as EngineModel).get(); +// } catch (error) { +// throw error; +// } +// }; + +// export const permissionsRemoveOne = async (model: any, id: string, input: NewPermissionInput, ctx: ApolloContext): Promise => { +// try { +// await model as EngineModel).updateOneCustom({ _id: id }, { $pull: { [`${input.permissionType}`]: input.permission } }, ctx); +// return model as EngineModel).get(); +// } catch (error) { +// throw error; +// } +// }; + +// export const publishOneGeneric = async (model: T, id: string, ctx: ApolloContext): Promise => { +// try { +// const missingInputs: string[] = []; +// // TODO THE CHECK OF THE INPUTS + +// if (missingInputs.length > 0) throw newError('missingInput', '400', missingInputs); + +// return editOneGeneric(model, id, { published: true, publicationDate: new Date() } as any, ctx); +// } catch (error) { +// throw error; +// } +// }; + +// export const unpublishOneGeneric = async (model: T, id: string, ctx: ApolloContext): Promise => { +// try { +// return editOneGeneric(model, id, { published: false } as any, ctx); +// } catch (error) { +// throw error; +// } +// }; diff --git a/lib/seed/engine/utils/streams/schemas/stream.components.ts b/lib/seed/engine/utils/streams/schemas/stream.components.ts new file mode 100644 index 0000000..93843b1 --- /dev/null +++ b/lib/seed/engine/utils/streams/schemas/stream.components.ts @@ -0,0 +1,41 @@ +import { registerEnumType } from 'type-graphql'; + +/* + ███████╗███╗ ██╗██╗ ██╗███╗ ███╗ + ██╔════╝████╗ ██║██║ ██║████╗ ████║ + █████╗ ██╔██╗ ██║██║ ██║██╔████╔██║ + ██╔══╝ ██║╚██╗██║██║ ██║██║╚██╔╝██║ + ███████╗██║ ╚████║╚██████╔╝██║ ╚═╝ ██║ + ╚══════╝╚═╝ ╚═══╝ ╚═════╝ ╚═╝ ╚═╝ +*/ + +export enum StreamOperationType { + insert = 'insert', + delete = 'delete', + replace = 'replace', + update = 'update', + hook = 'hook', +} +registerEnumType(StreamOperationType, { + name: 'StreamOperationType', +}); + +export enum PostHookStatus { + new = 'new', + inProcess = 'inProcess', + completed = 'completed', + error = 'error', + noAction = 'noAction', +} +registerEnumType(PostHookStatus, { + name: 'PostHookStatus', +}); + +/* + ██████╗ ██████╗ ███╗ ███╗██████╗ ██████╗ ███╗ ██╗███████╗███╗ ██╗████████╗███████╗ + ██╔════╝██╔═══██╗████╗ ████║██╔══██╗██╔═══██╗████╗ ██║██╔════╝████╗ ██║╚══██╔══╝██╔════╝ + ██║ ██║ ██║██╔████╔██║██████╔╝██║ ██║██╔██╗ ██║█████╗ ██╔██╗ ██║ ██║ ███████╗ + ██║ ██║ ██║██║╚██╔╝██║██╔═══╝ ██║ ██║██║╚██╗██║██╔══╝ ██║╚██╗██║ ██║ ╚════██║ + ╚██████╗╚██████╔╝██║ ╚═╝ ██║██║ ╚██████╔╝██║ ╚████║███████╗██║ ╚████║ ██║ ███████║ + ╚═════╝ ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═════╝ ╚═╝ ╚═══╝╚══════╝╚═╝ ╚═══╝ ╚═╝ ╚══════╝ +*/ diff --git a/lib/seed/engine/utils/streams/schemas/stream.schema.input.ts b/lib/seed/engine/utils/streams/schemas/stream.schema.input.ts new file mode 100644 index 0000000..3d76a60 --- /dev/null +++ b/lib/seed/engine/utils/streams/schemas/stream.schema.input.ts @@ -0,0 +1,11 @@ +import { GetManyArgs } from '@seed/graphql/Request'; +import { Field, ArgsType } from 'type-graphql'; + +@ArgsType() +export class StreamArgs extends GetManyArgs { + @Field({ nullable: true }) + documentKey?: string; + + @Field({ nullable: true }) + hookStatus?: string; +} diff --git a/lib/seed/engine/utils/streams/schemas/stream.schema.ts b/lib/seed/engine/utils/streams/schemas/stream.schema.ts new file mode 100644 index 0000000..7637daa --- /dev/null +++ b/lib/seed/engine/utils/streams/schemas/stream.schema.ts @@ -0,0 +1,50 @@ +import { AsyncHooksService } from '@lib/__hooks'; +import { IEngineSchema, MetaBy, MetaPermissions } from '@seed/engine/EngineSchema'; +import { StreamOperationType, PostHookStatus } from '@seed/services/change-stream/change-stream.components'; +import { ModelCollectionEnum } from '@src/__indexes/__collections'; +import GraphQLJSON from 'graphql-type-json'; +import { InputType, ObjectType, Field } from 'type-graphql'; + +@ObjectType() +@InputType() +export class StreamBaseSchema { + @Field(() => StreamOperationType) + operation: StreamOperationType; + + @Field(() => String) + collection: ModelCollectionEnum | keyof AsyncHooksService | string; + + @Field({ nullable: true }) + documentKey?: string; + + @Field(() => PostHookStatus) + hookStatus: PostHookStatus; + + @Field({ nullable: true }) + logMessage?: string; + + @Field(() => GraphQLJSON, { nullable: true }) + insertedValues?: any; + @Field(() => GraphQLJSON, { nullable: true }) + updatedValues?: any; + @Field(() => GraphQLJSON, { nullable: true }) + beforeUpdateValues?: any; + + logInfo?: any; +} + +@ObjectType() +export class StreamDBInterfaceSchema extends StreamBaseSchema {} + +@ObjectType() +export class StreamDBSchema extends StreamDBInterfaceSchema {} + +@ObjectType({ implements: IEngineSchema }) +export class StreamSchema extends StreamDBSchema implements IEngineSchema { + _id: string; + organisationId?: string | undefined; + by?: MetaBy | undefined; + permissions: MetaPermissions; + createdAt: Date; + updatedAt: Date; +} diff --git a/lib/seed/engine/utils/streams/stream.model.ts b/lib/seed/engine/utils/streams/stream.model.ts new file mode 100644 index 0000000..e6a1209 --- /dev/null +++ b/lib/seed/engine/utils/streams/stream.model.ts @@ -0,0 +1,49 @@ +import { ObjectType } from 'type-graphql'; + +import { Permission } from '@seed/interfaces/permission'; + +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { StreamDBInterfaceSchema, StreamDBSchema, StreamSchema } from './schemas/stream.schema'; +import { EngineModel } from '@seed/engine/EngineModel'; +import { IEngineSchema } from '@seed/engine/EngineSchema'; +import { ModelCollectionEnum } from '@src/__indexes/__collections'; +import { plainToClass } from 'class-transformer'; +import { CustomSearchEngine } from '@seed/services/database/DBRequestService'; +import { StreamArgs } from './schemas/stream.schema.input'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.admin], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class StreamEngineModel extends EngineModel { + public constructor(input?: StreamDBInterfaceSchema & Partial) { + const dataInit = plainToClass(StreamDBSchema, input || {}); + super({ + // ...init, + collectionName: ModelCollectionEnum.streams, + permissions: permissions, + dataInit, + }); + } + + plainToClass(plain: any): StreamSchema | StreamSchema[] { + return plainToClass(StreamSchema, plain); + } + + searchEngine(): CustomSearchEngine { + const sEngine: CustomSearchEngine = { + documentKey: { + operation: '$eq', + }, + hookStatus: { + operation: '$eq', + }, + }; + + return sEngine; + } +} diff --git a/lib/seed/engine/utils/streams/stream.resolver.ts b/lib/seed/engine/utils/streams/stream.resolver.ts new file mode 100644 index 0000000..e69de29 diff --git a/lib/seed/engine/utils/streams/stream.service.ts b/lib/seed/engine/utils/streams/stream.service.ts new file mode 100644 index 0000000..0aba53d --- /dev/null +++ b/lib/seed/engine/utils/streams/stream.service.ts @@ -0,0 +1,84 @@ +import { AsyncHooksService } from '@lib/__hooks'; +import { EngineModel } from '@seed/engine/EngineModel'; +import { IEngineSchema } from '@seed/engine/EngineSchema'; +import DB from '@seed/services/database/DBService'; +import { asyncHook } from '@seed/services/hooks/hooks.decorator'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { ModelCollectionEnum } from '@src/__indexes/__collections'; +import { v4 as uuid } from 'uuid'; +import { StreamOperationType, PostHookStatus } from './schemas/stream.components'; +import { StreamDBInterfaceSchema } from './schemas/stream.schema'; + +export const createChangeStream = async >(model: T, operation: StreamOperationType): Promise => { + let operationFunction; + + const changeStreamId = uuid(); + const base = { + _id: changeStreamId, + operation, + collection: model.collectionName, + documentKey: model._id, + hookStatus: PostHookStatus.new, + createdAt: new Date(), + updatedAt: new Date(), + }; + + switch (operation) { + case StreamOperationType.insert: + (base as any).insertedValues = model.get(); + operationFunction = 'afterCreate'; + break; + case StreamOperationType.update: + case StreamOperationType.replace: + (base as any).updatedValues = model.get(); + operationFunction = 'afterUpdate'; + break; + case StreamOperationType.delete: + (base as any).updatedValues = model.get(); + operationFunction = 'afterDelete'; + break; + default: + break; + } + + const dataToSave: StreamDBInterfaceSchema & IEngineSchema = { + ...base, + permissions: { + r: [AccountTypeEnum.admin], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], + }, + }; + await (await DB.getInstance()).db.collection(ModelCollectionEnum['streams']).insertOne(dataToSave); + + if (operationFunction && model[operationFunction] && (await asyncHook(dataToSave))) { + await model[operationFunction](); + + await (await DB.getInstance()).db + .collection(ModelCollectionEnum['streams']) + .updateOne({ _id: changeStreamId }, { $set: { hookStatus: PostHookStatus.completed } }); + } +}; + +export const createAsyncStream = async >(collection: keyof AsyncHooksService, data: any): Promise => { + const changeStreamId = uuid(); + const base = { + _id: changeStreamId, + operation: StreamOperationType.hook, + collection, + insertedValues: data, + hookStatus: PostHookStatus.new, + createdAt: new Date(), + updatedAt: new Date(), + }; + + const dataToSave: StreamDBInterfaceSchema & IEngineSchema = { + ...base, + permissions: { + r: [AccountTypeEnum.admin], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], + }, + }; + await (await DB.getInstance()).db.collection(ModelCollectionEnum['streams']).insertOne(dataToSave); +}; diff --git a/lib/seed/examples/__collections.ts b/lib/seed/examples/__collections.ts new file mode 100644 index 0000000..c455082 --- /dev/null +++ b/lib/seed/examples/__collections.ts @@ -0,0 +1,9 @@ +import { registerEnumType } from 'type-graphql'; + +export enum ModelCollectionEnum { + 'streams' = 'streams', + accounts = 'accounts', +} +registerEnumType(ModelCollectionEnum, { + name: 'ModelLoadersEnum', +}); diff --git a/lib/seed/examples/__cronjobs/serverless-cron.yaml b/lib/seed/examples/__cronjobs/serverless-cron.yaml new file mode 100644 index 0000000..e7bca20 --- /dev/null +++ b/lib/seed/examples/__cronjobs/serverless-cron.yaml @@ -0,0 +1,37 @@ +app: ${file(./package.json):name} +service: cronjob +package: + individually: false +provider: + name: aws + stage: ${opt:stage,'dev'} + runtime: nodejs12.x + environment: + AWS_compute: ${self:service}-${self:provider.stage}-compute + iamRoleStatements: + - Effect: Allow + Action: + - lambda:InvokeFunction + Resource: '*' +plugins: + - serverless-webpack +custom: + stage: ${opt:stage, self:provider.stage} + webpack: + webpackConfig: './lib/seed/webpack.config.js' # Name of webpack configuration file + includeModules: + forceExclude: + - aws-sdk + - puppeteer + +functions: + treatRemainingDaysCron: + handler: lib/__cronjobs/handler.cron1 + timeout: 300 # optional, in seconds, default is 6 + events: + - schedule: rate(1 day) + jobNotificationsCron: + handler: lib/__cronjobs/handler.cron2 + timeout: 300 + events: + - schedule: rate(1 day) diff --git a/lib/seed/examples/__hooks/functions/example.hooks.ts b/lib/seed/examples/__hooks/functions/example.hooks.ts new file mode 100644 index 0000000..fa5d966 --- /dev/null +++ b/lib/seed/examples/__hooks/functions/example.hooks.ts @@ -0,0 +1,8 @@ +import { ApolloContext } from '@seed/interfaces/context'; +import { promiseAll, sleep } from '@seed/helpers/Utils'; + +export const afterCheckoutEvent = async (data: any, ctx: ApolloContext): Promise => { + const promises: Promise[] = []; + + await promiseAll(promises); +}; diff --git a/lib/seed/examples/__hooks/index.ts b/lib/seed/examples/__hooks/index.ts new file mode 100644 index 0000000..e6cfaec --- /dev/null +++ b/lib/seed/examples/__hooks/index.ts @@ -0,0 +1,35 @@ +import { asyncHookDecorator } from '@seed/services/hooks/hooks.decorator'; +import { args } from 'chrome-aws-lambda'; +import { afterCheckoutEvent } from './functions/example.hooks'; +import { afterOrderCancel, afterOrdersMarkOneAsPaid, afterOrdersMarkOneAsUnPaid } from './functions/orders.hooks'; + +@asyncHookDecorator() +export class AsyncHooksService { + /* [MODULES - SHOP] */ + + public async afterCheckout(...args: any): Promise { + console.log('[NO ACTION] - afterCheckout'); + } + + public async afterOrderCancel(...args: Parameters): Promise { + return afterOrderCancel(...args); + } + + public async afterOrdersMarkOneAsPaid(...args: Parameters): Promise { + return afterOrdersMarkOneAsPaid(...args); + } + + public async afterOrdersMarkOneAsUnPaid(...args: Parameters): Promise { + return afterOrdersMarkOneAsUnPaid(...args); + } + + public async afterOrdersReimbursed(...args: Parameters): Promise { + console.log('[NO ACTION] - afterOrdersReimbursed'); + } + + /* [MODULES - EVENTS] */ + + public async afterCheckoutEvent(...args: Parameters): Promise { + return afterCheckoutEvent(...args); + } +} diff --git a/lib/seed/examples/app-rest.ts b/lib/seed/examples/app-rest.ts new file mode 100644 index 0000000..0dd8ea2 --- /dev/null +++ b/lib/seed/examples/app-rest.ts @@ -0,0 +1,11 @@ +import 'reflect-metadata'; + +import serverless from 'serverless-http'; +import express from 'express'; +const app = express(); + +app.get('/', function(req, res) { + res.send('Hello World!'); +}); + +module.exports.handler = serverless(app); diff --git a/lib/seed/examples/config.ts b/lib/seed/examples/config.ts new file mode 100644 index 0000000..bf29051 --- /dev/null +++ b/lib/seed/examples/config.ts @@ -0,0 +1,34 @@ +import ErrorSettingsModel from '@services/module-cms/functions/error-settings/errors-settings.model'; + +/* + ███████╗███╗ ██╗██╗ ██╗███╗ ███╗ + ██╔════╝████╗ ██║██║ ██║████╗ ████║ + █████╗ ██╔██╗ ██║██║ ██║██╔████╔██║ + ██╔══╝ ██║╚██╗██║██║ ██║██║╚██╔╝██║ + ███████╗██║ ╚████║╚██████╔╝██║ ╚═╝ ██║ + ╚══════╝╚═╝ ╚═══╝ ╚═════╝ ╚═╝ ╚═╝ +*/ + +/* + ██████╗ ██████╗ ███╗ ███╗██████╗ ██████╗ ███╗ ██╗███████╗███╗ ██╗████████╗███████╗ + ██╔════╝██╔═══██╗████╗ ████║██╔══██╗██╔═══██╗████╗ ██║██╔════╝████╗ ██║╚══██╔══╝██╔════╝ + ██║ ██║ ██║██╔████╔██║██████╔╝██║ ██║██╔██╗ ██║█████╗ ██╔██╗ ██║ ██║ ███████╗ + ██║ ██║ ██║██║╚██╔╝██║██╔═══╝ ██║ ██║██║╚██╗██║██╔══╝ ██║╚██╗██║ ██║ ╚════██║ + ╚██████╗╚██████╔╝██║ ╚═╝ ██║██║ ╚██████╔╝██║ ╚████║███████╗██║ ╚████║ ██║ ███████║ + ╚═════╝ ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═════╝ ╚═╝ ╚═══╝╚══════╝╚═╝ ╚═══╝ ╚═╝ ╚══════╝ +*/ + +export enum NotificationEnum { + 'notification' = 'notification', +} + +export const ErrorsConfig: { + [key: string]: { + message: string; + translations?: ErrorSettingsModel; + }; +} = { + code: { + message: 'message', + }, +}; diff --git a/lib/seed/examples/domains.yaml b/lib/seed/examples/domains.yaml new file mode 100644 index 0000000..f7eeee6 --- /dev/null +++ b/lib/seed/examples/domains.yaml @@ -0,0 +1,17 @@ +## Instal ## +## run : node ./lib/seed/devops/deploy-domains.js +## Carefull - Must be us-east-1 for certificate (created beforehand) + +domains: + production: XXXX.com + staging: staging-XXXX.com + dev: dev-XXXX.com +customDomain: + basePath: '${self:service}' + domainName: ${self:custom.domains.${self:custom.stage}} + stage: '${self:custom.stage}' + createRoute53Record: true + certificateName: XXXX + certificateArn: arn:aws:acm:us-east-1:555646219416:certificate/07e83105-1554-4e3d-b968-0a34d77911ad + endpointType: 'edge' + securityPolicy: tls_1_2 diff --git a/lib/seed/examples/env/__env.accounts.createAdmin.json b/lib/seed/examples/env/__env.accounts.createAdmin.json new file mode 100644 index 0000000..ce8c293 --- /dev/null +++ b/lib/seed/examples/env/__env.accounts.createAdmin.json @@ -0,0 +1,20 @@ +{ + "_id" : "23wzbPpt4OWXtqs92dAaNvrue9v1", + "d" : [ + "admin" + ], + "email" : "admin@xxxx", + "r" : [ + "admin" + ], + "types" : [ + "admin" + ], + "w" : [ + "admin" + ], + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "firstName" : "Admin", + "lastName" : "Project", +} \ No newline at end of file diff --git a/lib/seed/examples/env/__env.firebase.import.json b/lib/seed/examples/env/__env.firebase.import.json new file mode 100644 index 0000000..ce022e6 --- /dev/null +++ b/lib/seed/examples/env/__env.firebase.import.json @@ -0,0 +1,170 @@ +{ + "_id" : "b92eebfa-2bd9-4976-9a6d-2fba6bad14fb", + "key" : "FIREBASE_project_id", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "d" : [ + "admin", + ], + "r" : [ + "admin", + ], + "w" : [ + "admin", + ] +}, +{ + "_id" : "f6f8f6d6-25a4-4ce7-a00b-8b0a7188a5cf", + "key" : "FIREBASE_private_key_id", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "d" : [ + "admin", + ], + "r" : [ + "admin", + ], + "w" : [ + "admin", + ] +}, +{ + "_id" : "00698ef9-7dc3-4942-973d-ac5823ca6399", + "key" : "FIREBASE_private_key", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin", + ], + "w" : [ + "admin", + ], + "d" : [ + "admin", + ], +}, +{ + "_id" : "829f390c-ca20-4dec-a679-4f2117fee710", + "key" : "FIREBASE_client_email", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "d" : [ + "admin", + ], + "r" : [ + "admin", + ], + "w" : [ + "admin", + ] +}, +{ + "_id" : "7f86b770-f6ce-4c99-a6ad-7eda6c3eac04", + "key" : "FIREBASE_client_id", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "d" : [ + "admin", + ], + "r" : [ + "admin", + ], + "w" : [ + "admin", + ] +}, +{ + "_id" : "262c6d6f-ff34-46b4-9639-f8baa75186ce", + "key" : "FIREBASE_auth_uri", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin", + ], + "w" : [ + "admin", + ], + "d" : [ + "admin", + ], +}, +{ + "_id" : "b34ec29f-02c1-4699-ab08-83dd920bce7a", + "key" : "FIREBASE_token_uri", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin", + ], + "w" : [ + "admin", + ], + "d" : [ + "admin", + ], +}, +{ + "_id" : "35409f42-f87f-47ae-8505-2328f98a3979", + "key" : "FIREBASE_auth_provider_x509_cert_url", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin", + ], + "w" : [ + "admin", + ], + "d" : [ + "admin", + ], +}, +{ + "_id" : "fbb3f410-6717-49af-8004-75c9f110e7db", + "key" : "FIREBASE_client_x509_cert_url", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin", + ], + "w" : [ + "admin", + ], + "d" : [ + "admin", + ], +}, +{ + "_id" : "34e7ad98-3c5b-4288-802e-57f6f67a2c9f", + "key" : "FIREBASE_apiKey", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin", + ], + "w" : [ + "admin", + ], + "d" : [ + "admin", + ], +}, \ No newline at end of file diff --git a/lib/seed/examples/env/__env.general.import.json b/lib/seed/examples/env/__env.general.import.json new file mode 100644 index 0000000..5d2c27f --- /dev/null +++ b/lib/seed/examples/env/__env.general.import.json @@ -0,0 +1,169 @@ +{ + "_id" : "28ade57d-697b-4cac-9e50-62cd4a8c8e61", + "key" : "FILE_allowedMime", + "value" :"image/jpeg,image/png,image/gif,application/pdf,image/svg+xml,application/xml,application/zip", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "d" : [ + "admin", + ], + "r" : [ + "admin", + ], + "w" : [ + "admin", + ] +}, +{ + "_id" : "6f106c8f-df69-403d-b92b-6bba314ce6ef", + "key" : "TEST_USER_EMAIL", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "d" : [ + "admin", + ], + "r" : [ + "admin", + ], + "w" : [ + "admin", + ] +}, +{ + "_id" : "cfc32155-cd1c-4d5e-bfa9-425ef27065ce", + "key" : "TEST_ADMIN_EMAIL", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "d" : [ + "admin", + ], + "r" : [ + "admin", + ], + "w" : [ + "admin", + ] +}, +{ + "_id" : "8c4c28fb-d89d-434c-a962-f4c43dbda68d", + "key" : "STRIPE_SKEY", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin", + ], + "w" : [ + "admin", + ], + "d" : [ + "admin", + ], +}, +{ + "_id" : "dec94533-a457-4c66-9ead-4e754243ec89", + "key" : "STRIPE_PUBLIC_KEY", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin", + ], + "w" : [ + "admin", + ], + "d" : [ + "admin", + ], +}, +{ + "_id" : "870d1674-367a-4937-9d66-d36e889d4953", + "key" : "STRIPE_CURRENCY", + "value" :"usd", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin", + ], + "w" : [ + "admin", + ], + "d" : [ + "admin", + ], +}, +{ + "_id" : "5ed19459-1b4b-462b-88fb-5fbb2d698229", + "key" : "STRIPE_ORDERDESCRIPTION", + "value" :"You order from our webplatform", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin", + ], + "w" : [ + "admin", + ], + "d" : [ + "admin", + ], +}, +{ + "_id" : "354b4491-497f-4e3f-9569-06863e19fadd", + "key" : "MJ_APIKEY_PUBLIC", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin", + ], + "w" : [ + "admin", + ], + "d" : [ + "admin", + ], +}, +{ + "_id" : "2e69bc94-28c9-400e-bd77-1714fb4f8300", + "key" : "MJ_APIKEY_PRIVATE", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin", + ], + "w" : [ + "admin", + ], + "d" : [ + "admin", + ], +},{ + "_id" : "5320e684-77af-4432-b229-60d63825a823", + "key" : "GOOGLE_MAP_apiKey", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin", + ], + "w" : [ + "admin", + ], + "d" : [ + "admin", + ], +} \ No newline at end of file diff --git a/lib/seed/examples/env/__env.mailsettings.import.json b/lib/seed/examples/env/__env.mailsettings.import.json new file mode 100644 index 0000000..8099fa8 --- /dev/null +++ b/lib/seed/examples/env/__env.mailsettings.import.json @@ -0,0 +1,219 @@ +{ + "_id" : "04e48fa2-714f-4d7f-ae17-c791ccf6e5bb", + "key" : "resetPassword", + "templateId" : { + "en" : "", + "fr" : "", + "nl" : "" + }, + "fromEmail" : "no-reply@makeit-studio.com", + "replyToEmail" : "software@makeit-studio.com", + "fromName" : "Make it | Studio", + "subject" : { + "en" : "You requested a new password", + "fr" : "Vous avez demandé un nouveau mot de passe", + "nl" : "You requested a new password" + }, + "body" : { + "en" : "

Here is the password reset link : ${resetPasswordLink}

", + "fr" : "

Voici le lien pour réinitialiser votre mot de passe : ${resetPasswordLink}

", + "nl" : "

Here is the password reset link : ${resetPasswordLink}

" + }, + "availableFields" : [ + "firstName", + "lastName", + "email", + "resetPasswordLink" + ], + "custom" : false, + "type" : "emails", + "cci" : [ + + ], + "r" : [ + "admin" + ], + "w" : [ + "admin" + ], + "d" : [ + "admin" + ], + "createdAt" : ISODate(), + "updatedAt" : ISODate(), +}, +{ + "_id" : "0f245c6c-e39a-465f-9374-e2268a0cbf83", + "key" : "adminAccountAddOne", + "templateId" : { + "en" : "", + "fr" : "", + "nl" : "" + }, + "fromEmail" : "no-reply@makeit-studio.com", + "replyToEmail" : "software@makeit-studio.com", + "fromName" : "Make it | Studio", + "subject" : { + "en" : "The administrator of [PROJECT] created an account for you", + "fr" : "L'administrateur de [PROJECT] a crée un compte pour vous!", + "nl" : "The administrator of [PROJECT] created an account for you" + }, + "body" : { + "en" : "

Please setup your password by clicking this link : ${resetPasswordLink}

", + "fr" : "

Pouvez-vous configurer un mot de passe en cliquant sur ce lien ? : ${resetPasswordLink}

", + "nl" : "

Please setup your password by clicking this link : ${resetPasswordLink}

" + }, + "availableFields" : [ + "firstName", + "lastName", + "email", + "resetPasswordLink" + ], + "custom" : false, + "type" : "emails", + "cci" : [ + + ], + "r" : [ + "admin" + ], + "w" : [ + "admin" + ], + "d" : [ + "admin" + ], + "createdAt" : ISODate(), + "updatedAt" : ISODate(), +}, +{ + "_id" : "16496dfc-5935-495b-a352-52efe158d55c", + "key" : "organisationsAddOneAccount", + "templateId" : { + "en" : "", + "fr" : "", + "nl" : "" + }, + "fromEmail" : "no-reply@makeit-studio.com", + "replyToEmail" : "software@makeit-studio.com", + "fromName" : "Make it | Studio", + "subject" : { + "en" : "The administrator of [ORG] created an account for you", + "fr" : "L'administrateur de l'organisation [ORG] a crée un compte pour vous!", + "nl" : "The administrator of [ORG] created an account for you" + }, + "body" : { + "en" : "

Please setup your password by clicking this link : ${resetPasswordLink}

", + "fr" : "

Pouvez-vous configurer un mot de passe en cliquant sur ce lien ? : ${resetPasswordLink}

", + "nl" : "

Please setup your password by clicking this link : ${resetPasswordLink}

" + }, + "availableFields" : [ + "firstName", + "lastName", + "email", + "resetPasswordLink" + ], + "custom" : false, + "type" : "emails", + "cci" : [ + + ], + "r" : [ + "admin" + ], + "w" : [ + "admin" + ], + "d" : [ + "admin" + ], + "createdAt" : ISODate(), + "updatedAt" : ISODate(), +}, +{ + "_id" : "16496dfc-5875-495b-axx2-52efe158d55c", + "key" : "emailCodeSignIn", + "templateId" : { + "en" : "", + "fr" : "", + "nl" : "" + }, + "fromEmail" : "no-reply@makeit-studio.com", + "replyToEmail" : "software@makeit-studio.com", + "fromName" : "Make it | Studio", + "subject" : { + "en" : "Here is your code to login : ${code}", + "fr" : "Voici votre code d'accès: ${code}", + "nl" : "Here is your code to login : ${code}" + }, + "body" : { + "en" : "Here is your code to login : ${code}", + "fr" : "Voici votre code d'accès: ${code}", + "nl" : "Here is your code to login : ${code}" + }, + "availableFields" : [ + "firstName", + "code" + ], + "custom" : false, + "type" : "emails", + "cci" : [ + + ], + "r" : [ + "admin" + ], + "w" : [ + "admin" + ], + "d" : [ + "admin" + ], + "createdAt" : ISODate(), + "updatedAt" : ISODate(), +}, +{ + "_id" : "04e48fa2-714f-4d7f-ae17-c791ccf6e5bb", + "key" : "magicLink", + "templateId" : { + "en" : "", + "fr" : "", + "nl" : "" + }, + "subject" : { + "en" : "Your magic link is ready", + "fr" : "Votre demande de lien d'accès", + "nl" : "Your magic link is ready" + }, + "body" : { + "en" : "Here is your link to login : ${magicLink}", + "fr" : "Voici votre lien d'accès: ${magicLink}", + "nl" : "Here is your link to login : ${magicLink}" + }, + "fromEmail" : "no-reply@makeit-studio.com", + "replyToEmail" : "software@makeit-studio.com", + "fromName" : "Make it | Studio", + "availableFields" : [ + "firstName", + "lastName", + "email", + "magicLink" + ], + "custom" : false, + "type" : "emails", + "cci" : [ + + ], + "r" : [ + "admin" + ], + "w" : [ + "admin" + ], + "d" : [ + "admin" + ], + "createdAt" : ISODate(), + "updatedAt" : ISODate(), +} + diff --git a/lib/seed/examples/env/__env.notifications.import.json b/lib/seed/examples/env/__env.notifications.import.json new file mode 100644 index 0000000..ca30cc4 --- /dev/null +++ b/lib/seed/examples/env/__env.notifications.import.json @@ -0,0 +1,51 @@ +{ + "_id" : "28ade57d-df69-4cac-9e50-62cd4a8c8e61", + "key" : "TWILIO_FROM_NUMBER", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "d" : [ + "admin", + ], + "r" : [ + "admin", + ], + "w" : [ + "admin", + ] +}, +{ + "_id" : "28ade57d-df69-403d-b92b-6bba314ce6ef", + "key" : "TWILIO_ACCOUNT_SID", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "d" : [ + "admin", + ], + "r" : [ + "admin", + ], + "w" : [ + "admin", + ] +}, +{ + "_id" : "cfc32155-b92b-4d5e-bfa9-425ef27065ce", + "key" : "TWILIO_AUTH_TOKEN", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "d" : [ + "admin", + ], + "r" : [ + "admin", + ], + "w" : [ + "admin", + ] +}, diff --git a/lib/seed/examples/list.import.json b/lib/seed/examples/list.import.json new file mode 100644 index 0000000..ef88703 --- /dev/null +++ b/lib/seed/examples/list.import.json @@ -0,0 +1,47 @@ +mutation listsAddOne { + listsAddOne1: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_2.svg",fr:"combined-shape_2.svg",nl:"combined-shape_2.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_2.svg"}}){_id} + listsAddOne2: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_3.svg",fr:"combined-shape_3.svg",nl:"combined-shape_3.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_3.svg"}}){_id} + listsAddOne3: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_4.svg",fr:"combined-shape_4.svg",nl:"combined-shape_4.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_4.svg"}}){_id} + listsAddOne4: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_5.svg",fr:"combined-shape_5.svg",nl:"combined-shape_5.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_5.svg"}}){_id} + listsAddOne5: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_6.svg",fr:"combined-shape_6.svg",nl:"combined-shape_6.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_6.svg"}}){_id} + listsAddOne6: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_7.svg",fr:"combined-shape_7.svg",nl:"combined-shape_7.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_7.svg"}}){_id} + listsAddOne7: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_8.svg",fr:"combined-shape_8.svg",nl:"combined-shape_8.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_8.svg"}}){_id} + listsAddOne8: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_9.svg",fr:"combined-shape_9.svg",nl:"combined-shape_9.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_9.svg"}}){_id} + listsAddOne9: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_10.svg",fr:"combined-shape_10.svg",nl:"combined-shape_10.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_10.svg"}}){_id} + listsAddOne11: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_11.svg",fr:"combined-shape_11.svg",nl:"combined-shape_11.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_11.svg"}}){_id} + listsAddOne12: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_12.svg",fr:"combined-shape_12.svg",nl:"combined-shape_12.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_12.svg"}}){_id} + listsAddOne13: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_13.svg",fr:"combined-shape_13.svg",nl:"combined-shape_13.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_13.svg"}}){_id} + listsAddOne14: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_14.svg",fr:"combined-shape_14.svg",nl:"combined-shape_14.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_14.svg"}}){_id} + listsAddOne15: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_15.svg",fr:"combined-shape_15.svg",nl:"combined-shape_15.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_15.svg"}}){_id} + listsAddOne16: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_16.svg",fr:"combined-shape_16.svg",nl:"combined-shape_16.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_16.svg"}}){_id} + listsAddOne17: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_17.svg",fr:"combined-shape_17.svg",nl:"combined-shape_17.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_17.svg"}}){_id} + listsAddOne18: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_18.svg",fr:"combined-shape_18.svg",nl:"combined-shape_18.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_18.svg"}}){_id} + listsAddOne19: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_19.svg",fr:"combined-shape_19.svg",nl:"combined-shape_19.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_19.svg"}}){_id} + listsAddOne21: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_20.svg",fr:"combined-shape_20.svg",nl:"combined-shape_20.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_20.svg"}}){_id} + listsAddOne22: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_21.svg",fr:"combined-shape_21.svg",nl:"combined-shape_21.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_21.svg"}}){_id} + listsAddOne23: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_22.svg",fr:"combined-shape_22.svg",nl:"combined-shape_22.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_22.svg"}}){_id} + listsAddOne24: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_23.svg",fr:"combined-shape_23.svg",nl:"combined-shape_23.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_23.svg"}}){_id} + listsAddOne25: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_24.svg",fr:"combined-shape_24.svg",nl:"combined-shape_24.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_24.svg"}}){_id} + listsAddOne26: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_25.svg",fr:"combined-shape_25.svg",nl:"combined-shape_25.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_25.svg"}}){_id} + listsAddOne27: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_26.svg",fr:"combined-shape_26.svg",nl:"combined-shape_26.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_26.svg"}}){_id} + listsAddOne28: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_27.svg",fr:"combined-shape_27.svg",nl:"combined-shape_27.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_27.svg"}}){_id} + listsAddOne29: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_28.svg",fr:"combined-shape_28.svg",nl:"combined-shape_28.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_28.svg"}}){_id} + listsAddOne31: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_29.svg",fr:"combined-shape_29.svg",nl:"combined-shape_29.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_29.svg"}}){_id} + listsAddOne32: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_30.svg",fr:"combined-shape_30.svg",nl:"combined-shape_30.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_30.svg"}}){_id} + listsAddOne33: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape_31.svg",fr:"combined-shape_31.svg",nl:"combined-shape_31.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape_31.svg"}}){_id} + listsAddOne34: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape-copy-2.svg",fr:"combined-shape-copy-2.svg",nl:"combined-shape-copy-2.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape-copy-2.svg"}}){_id} + listsAddOne35: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape-copy-3.svg",fr:"combined-shape-copy-3.svg",nl:"combined-shape-copy-3.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape-copy-3.svg"}}){_id} + listsAddOne36: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape-copy-4.svg",fr:"combined-shape-copy-4.svg",nl:"combined-shape-copy-4.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape-copy-4.svg"}}){_id} + listsAddOne37: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape-copy-5.svg",fr:"combined-shape-copy-5.svg",nl:"combined-shape-copy-5.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape-copy-5.svg"}}){_id} + listsAddOne38: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape-copy-6.svg",fr:"combined-shape-copy-6.svg",nl:"combined-shape-copy-6.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape-copy-6.svg"}}){_id} + listsAddOne39: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape-copy-7.svg",fr:"combined-shape-copy-7.svg",nl:"combined-shape-copy-7.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape-copy-7.svg"}}){_id} + listsAddOne41: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape-copy-8.svg",fr:"combined-shape-copy-8.svg",nl:"combined-shape-copy-8.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape-copy-8.svg"}}){_id} + listsAddOne42: listsAddOne(input:{ressourceType:equipment,title:{en:"combined-shape.svg",fr:"combined-shape.svg",nl:"combined-shape.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/combined-shape.svg"}}){_id} + listsAddOne43: listsAddOne(input:{ressourceType:feature,title:{en:"group.svg",fr:"group.svg",nl:"group.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/group.svg"}}){_id} + listsAddOne44: listsAddOne(input:{ressourceType:feature,title:{en:"shape_2.svg",fr:"shape_2.svg",nl:"shape_2.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/shape_2.svg"}}){_id} + listsAddOne45: listsAddOne(input:{ressourceType:feature,title:{en:"shape_3.svg",fr:"shape_3.svg",nl:"shape_3.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/shape_3.svg"}}){_id} + listsAddOne46: listsAddOne(input:{ressourceType:feature,title:{en:"shape_4.svg",fr:"shape_4.svg",nl:"shape_4.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/shape_4.svg"}}){_id} + listsAddOne47: listsAddOne(input:{ressourceType:feature,title:{en:"shape_5.svg",fr:"shape_5.svg",nl:"shape_5.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/shape_5.svg"}}){_id} + listsAddOne48: listsAddOne(input:{ressourceType:feature,title:{en:"shape_6.svg",fr:"shape_6.svg",nl:"shape_6.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/shape_6.svg"}}){_id} + listsAddOne49: listsAddOne(input:{ressourceType:feature,title:{en:"shape.svg",fr:"shape.svg",nl:"shape.svg"},thumbnail:{large:"https://workinflex-uploads-production.s3.eu-central-1.amazonaws.com/shape.svg"}}){_id} + } \ No newline at end of file diff --git a/lib/seed/graphql/AccessService.ts b/lib/seed/graphql/AccessService.ts new file mode 100644 index 0000000..e26ced9 --- /dev/null +++ b/lib/seed/graphql/AccessService.ts @@ -0,0 +1,150 @@ +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { newError } from '@seed/helpers/Error'; + +export const checkPermissions = (ressource: any, account: any | null, type: 'c' | 'r' | 'w' | 'd'): boolean => { + // Adding the public by default + let perm: any[] = [AccountTypeEnum.public]; + + if (account && account.types) { + /* + * Verify if admin + */ + if (account.types.includes(AccountTypeEnum.admin)) return true; + + /* + * Verify if organisation type of access + */ + if ( + ressource.organisationId && + account.organisationIds && + !account.organisationIds.includes(ressource.organisationId) && + !ressource.r.includes(AccountTypeEnum.public) + ) + throw newError(2100, { allowed: ressource.organisationId, you: account.organisationIds }); + + /* + * Add the account id and type + */ + perm.push(account._id); + perm = perm.concat(account.types); + } + + /* + * Verify on the ressource level + */ + + const permissions = ressource[type]; + + // if there is no permission on the ressource, return true + if (!permissions) return true; + + let hasPerm = false; + + // Verifying if it matches + for (let index = 0; index < perm.length; index++) { + const element = perm[index]; + if (permissions.includes(element)) { + hasPerm = true; + break; + } + } + + if (!hasPerm) throw newError(2000, { allowed: permissions, you: account }); + + return true; +}; + +export const checkOrganisationPermissions = (ressource: any, organisationId: string): boolean => { + if (organisationId == ressource.organisationId || organisationId == ressource._id) return true; + throw newError(2000, { allowedOrgId: ressource.organisationId, youOrgId: organisationId }); +}; + +export const addPermissions = (ressource: any, type: ('r' | 'w' | 'd')[], ids: (string | string)[]): void => { + for (let index = 0; index < type.length; index++) { + const t = type[index]; + ressource[t] = ressource[t].concat(ids); + } +}; + +export const addPermissionToQuery = (account: any | null, query: 'get' | 'update' | 'delete', params: any): any => { + let types: any[] = ['public']; + + if (account && account.types) { + /* + * Verify if admin, no need to add the query filters + */ + if (account.types.includes(AccountTypeEnum.admin)) return params; + + /* + * Verify if organisation type of access + */ + + // if (account.organisationIds) { + // params.organisationId = { $in: account.organisationIds }; + // } + + /* + * Add the account id and type + */ + + types.push(account._id); + types = types.concat(account.types); + } + + if (!params.$and) params.$and = []; + + switch (query) { + default: + case 'get': + params.$and.push({ r: { $in: types } }); + break; + case 'update': + params.$and.push({ w: { $in: types } }); + break; + case 'delete': + params.$and.push({ d: { $in: types } }); + break; + } + + return params; +}; + +export const addOrganisationToQuery = (account: any | null, query: 'get' | 'update' | 'delete', params: any): any => { + let types: any[] = ['public']; + + if (account && account.types) { + /* + * Verify if admin, no need to add the query filters + */ + if (account.types.includes(AccountTypeEnum.admin)) return params; + + /* + * Verify if organisation type of access + */ + if (account.organisationIds) { + params.organisationId = { $in: account.organisationIds }; + } + + /* + * Add the account id and type + */ + + types.push(account._id); + types = types.concat(account.types); + } + + switch (query) { + default: + case 'get': + params = { ...params, $or: [{ r: { $in: types } }] }; + break; + case 'update': + params = { ...params, $or: [{ w: { $in: types } }] }; + break; + case 'delete': + params = { ...params, $or: [{ d: { $in: types } }] }; + break; + } + + return params; +}; diff --git a/lib/seed/graphql/BaseGraphModel.ts b/lib/seed/graphql/BaseGraphModel.ts new file mode 100644 index 0000000..cb6b704 --- /dev/null +++ b/lib/seed/graphql/BaseGraphModel.ts @@ -0,0 +1,828 @@ +import { v4 as uuid } from 'uuid'; +import { Collection, FilterQuery, UpdateQuery } from 'mongodb'; +import { classToPlain, Exclude, plainToClassFromExist } from 'class-transformer'; +import { Field, ID, ObjectType, Authorized } from 'type-graphql'; +import DB, { createSetRequest } from '@seed/services/database/DBService'; + +import { Permission } from '@seed/interfaces/permission'; +import { newError } from '@seed/helpers/Error'; + +import { parsePaginationOptions } from '@seed/helpers/Request'; +import { GetArgs, GetManyArgs } from './Request'; +import { addPermissions, addPermissionToQuery, checkPermissions, checkOrganisationPermissions } from './AccessService'; +import { ApolloContext } from '@seed/interfaces/context'; +import _ from 'lodash'; +import { StreamOperationType, PostHookStatus } from '@seed/services/change-stream/change-stream.components'; +import { asyncHook } from '@seed/services/hooks/hooks.decorator'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { promiseAll } from '@seed/helpers/Utils'; +import { EnginePathComponent } from '@seed/interfaces/components'; + +export interface BaseInterface { + _id: string; + + updatedAt: Date; + createdAt: Date; + + collectionName: string; +} +@ObjectType() +export abstract class BaseGraphModel implements BaseInterface { + @Exclude() + public collectionName: string; + @Exclude() + protected permissions: Permission; + @Exclude() + defaultSort: string; + + // eslint-disable-next-line @typescript-eslint/no-unused-vars + @Field(() => ID) + _id: string; + + @Field(() => ID, { nullable: true }) + organisationId?: string; + + @Field(() => ID, { nullable: true }) + createdBy?: string; + + @Field(() => ID, { nullable: true }) + updatedBy?: string; + + @Field(() => ID, { nullable: true }) + deletedBy?: string; + + @Field() + createdAt: Date; + + @Field() + updatedAt: Date; + + @Field(() => [String]) + public r: string[]; + + @Authorized('admin') + @Field(() => [String]) + public w: string[]; + + @Authorized('admin') + @Field(() => [String]) + public d: string[]; + + paths: EnginePathComponent[]; + + public constructor(init: { collectionName: string; permissions: Permission; defaultSort?: string }) { + this.collectionName = init.collectionName; + this.permissions = init.permissions; + + this.r = init.permissions.r; + this.w = init.permissions.w; + this.d = init.permissions.d; + + this.createdAt = new Date(); + this.updatedAt = new Date(); + + this.defaultSort = init.defaultSort || 'createdAt desc'; + } + + /* + * + * Model functions + * + */ + + public get(): any { + return classToPlain(this); + } + + public set(doc: any): void { + plainToClassFromExist(this, doc); + + if (doc._id) this._id = doc._id; + + if (doc.r) this.r = _.uniq(_.concat(this.r, doc.r)); + if (doc.w) this.w = _.uniq(_.concat(this.w, doc.w)); + if (doc.d) this.d = _.uniq(_.concat(this.d, doc.d)); + + if (doc.organisationId) this.organisationId = doc.organisationId; + + if (doc.createdAt) this.createdAt = doc.createdAt; + if (doc.updatedAt) this.updatedAt = doc.updatedAt; + } + + abstract searchOptions(): string[]; + abstract filterOptions(): string[]; + + searchEngine?(): any; + + async prehook?(): Promise; + async beforeCreate?(ctx?: ApolloContext | null): Promise; + async beforeUpdate?(ctx?: ApolloContext | null): Promise; + async beforeDelete?(ctx?: ApolloContext | null): Promise; + + async afterCreate?(changeStream: any): Promise; + async afterUpdate?(changeStream: any): Promise; + async afterDelete?(changeStream: any): Promise; + + /* + * + * DB Functions + * + */ + + public async db(): Promise> { + return await (await DB.getInstance()).db.collection(this.collectionName); + } + + public getPath(): EnginePathComponent[] { + if (this.paths) { + this.paths.push({ + ressourceModel: this.collectionName, + ressourceId: this._id, + }); + + return this.paths; + } else + return [ + { + ressourceModel: this.collectionName, + ressourceId: this._id, + }, + ]; + } + + public getQuery(query: FilterQuery, ctx: ApolloContext | null): any { + let finalQuery: any; + + if (ctx) { + const account = ctx.ctx.user; + const organisationId = ctx.ctx.organisationId; + + // We add the permission (on r field) to make sure that kind of user has access to the ressource + finalQuery = addPermissionToQuery(account, 'get', { ...query }); + + // We check if it comes from an organisation + // Check of the user's correct authorisation on that organisation comes from the middleware + + if (!ctx.ctx.noOrganisationCheck) { + if (organisationId) { + if (this.collectionName == 'accounts') finalQuery['organisationIds'] = organisationId; + else finalQuery['organisationId'] = { $eq: organisationId }; + } else { + // If admin & public query -> do not add this condition and let search every records) + if ( + account && + account.types && + !account.types.includes(AccountTypeEnum.public) && + !account.types.includes(AccountTypeEnum.admin) + ) { + finalQuery['organisationId'] = { $exists: false }; + } + } + } + } else { + finalQuery = { ...query }; + } + + return finalQuery; + } + + public async getMany(query: FilterQuery, pagination: GetArgs | undefined, ctx: ApolloContext | null): Promise { + const finalQuery = this.getQuery(query, ctx); + + if (!pagination) pagination = { limit: 100, skip: 0 }; + if (!pagination.sort) pagination.sort = this.defaultSort; + const paginationOption = parsePaginationOptions(pagination); + + try { + console.log('search request', JSON.stringify(finalQuery, null, 2)); + return await (await this.db()).find(finalQuery, paginationOption).toArray(); + } catch (error) { + return []; + } + } + + public async getCount(query: FilterQuery, ctx: ApolloContext | null): Promise { + const finalQuery = this.getQuery(query, ctx); + + try { + const result = (await (await this.db()).countDocuments(finalQuery)) as any; + console.log(result); + return result; + } catch (error) { + return 0; + } + } + + public async getAll(query: FilterQuery, ctx: ApolloContext | null): Promise { + const finalQuery = this.getQuery(query, ctx); + try { + return await (await this.db()).find(finalQuery).toArray(); + } catch (error) { + return []; + } + } + + public async getOne(query: FilterQuery, ctx: ApolloContext | null): Promise { + if (ctx) { + const account = ctx.ctx.user; + const organisationId = ctx.ctx.organisationId; + + const doc = await (await this.db()).findOne(query); + if (!doc) throw newError(404, query); + + this.set(doc); + + if (!ctx.ctx.noOrganisationCheck) { + if (organisationId) { + if (this.collectionName != 'accounts') checkOrganisationPermissions(this, organisationId); + } + } + + checkPermissions(this, account, 'r'); + + return this; + } else { + const doc = await (await this.db()).findOne(query); + if (!doc) throw newError(404, query); + + this.set(doc); + return this; + } + } + + public async saveOne(newData: Partial, ctx: ApolloContext | null, upsert = true): Promise { + if (ctx) { + const account = ctx.ctx.user; + const organisationId = ctx.ctx.organisationId; + + addPermissions(this, ['r', 'w', 'd'], [account._id]); + + if (organisationId) { + newData.organisationId = organisationId; + } + + newData.createdBy = account._id; + } + + this.set(newData); + + // if (!ctx) ctx = await createApolloContext(); + + try { + if (this.prehook) await this.prehook(); + if (this.beforeCreate) await this.beforeCreate(ctx); + } catch (error) { + throw error; + } + + const { _id, ...dataToSave } = this.get(); + const savedId = _id ? _id : uuid(); + + try { + const result = await (await this.db()).findOneAndUpdate( + { _id: savedId }, + { + $set: dataToSave, + $setOnInsert: { + _id: savedId, + }, + } as any, + { upsert, returnOriginal: false }, + ); + + if (result.ok == 0) throw newError(1000, result); + + this.set({ _id: savedId }); + + const changeStreamId = uuid(); + const changeStreamData = { + _id: changeStreamId, + operation: StreamOperationType.insert, + collection: this.collectionName, + documentKey: this._id, + insertedValues: this.get(), + hookStatus: PostHookStatus.new, + createdAt: new Date(), + updateAt: new Date(), + }; + await (await DB.getInstance()).db.collection('stream.changes').insertOne(changeStreamData); + + if (this.afterCreate && (await asyncHook(changeStreamData))) { + await this.afterCreate(changeStreamData); + + await (await DB.getInstance()).db + .collection('stream.changes') + .updateOne({ _id: changeStreamId }, { $set: { hookStatus: PostHookStatus.completed } }); + } + + return this; + } catch (error) { + throw newError(1000, error); + } + } + + public async saveMany(newDataArr: Partial[], ctx: ApolloContext | null): Promise { + // Initialize the change stream + const changeStreams: any[] = []; + const afterCreatePromises: any[] = []; + + const prehooksPromises = newDataArr.map(async (newData) => { + const clone = Object.assign(Object.create(Object.getPrototypeOf(this)), this); + if (ctx) { + const account = ctx.ctx.user; + const organisationId = ctx.ctx.organisationId; + + addPermissions(clone, ['r', 'w', 'd'], [account._id]); + + if (organisationId) { + newData.organisationId = organisationId; + } + + newData.createdBy = account._id; + } + + clone.set(newData); + + // if (!ctx) ctx = await createApolloContext(); + + if (clone.prehook) await clone.prehook(); + if (clone.beforeCreate) await clone.beforeCreate(ctx); + + const { _id, ...dataToSave } = clone.get(); + const savedId = _id ? _id : uuid(); + clone.set({ _id: savedId }); + + const changeStreamId = uuid(); + const changeStreamData = { + _id: changeStreamId, + operation: StreamOperationType.insert, + collection: clone.collectionName, + documentKey: clone._id, + insertedValues: clone.get(), + hookStatus: PostHookStatus.new, + createdAt: new Date(), + updateAt: new Date(), + }; + + changeStreams.push(changeStreamData); + + // eslint-disable-next-line @typescript-eslint/no-this-alias + afterCreatePromises.push(async () => { + if (clone.afterCreate) { + const res = await asyncHook(changeStreamData); + + if (res) { + await clone.afterCreate(changeStreamData); + + await (await DB.getInstance()).db + .collection('stream.changes') + .updateOne({ _id: changeStreamId }, { $set: { hookStatus: PostHookStatus.completed } }); + } + } + }); + + return clone.get(); + }); + + try { + const dataArr = await Promise.all(prehooksPromises); + await (await this.db()).insertMany(dataArr, { ordered: false }); + await (await DB.getInstance()).db.collection('stream.changes').insertMany(changeStreams, { ordered: false }); + await promiseAll(afterCreatePromises); + return dataArr; + } catch (error) { + console.error(error); + throw newError(1000, error); + } + } + + public async updateOne(query: FilterQuery, newData: Partial, ctx: ApolloContext | null): Promise { + try { + const dataThatHaveBeenChanged = createSetRequest('', { ...newData, updatedAt: new Date() }); + let dataBeforeChange: any; + + if (ctx) { + const account = ctx.ctx.user; + // Check if doc exists & correct permissions + if (!this._id) await this.getOne(query, ctx); + + dataBeforeChange = this.get(); + + dataThatHaveBeenChanged.updatedBy = account._id; + + this.set(dataThatHaveBeenChanged); + if (this.beforeUpdate) await this.beforeUpdate(ctx); + + checkPermissions(this, account, 'w'); + + const result = await (await this.db()).findOneAndUpdate( + query, + { $set: dataThatHaveBeenChanged }, + { upsert: false, returnOriginal: false }, + ); + if (result.ok != 1) throw newError(1000, result); + + this.set(result.value); + } else { + if (!this._id) await this.getOne(query, null); + dataBeforeChange = this.get(); + + this.set(dataThatHaveBeenChanged); + if (this.beforeUpdate) await this.beforeUpdate(); + + const result = await (await this.db()).findOneAndUpdate( + query, + { $set: dataThatHaveBeenChanged }, + { upsert: false, returnOriginal: false }, + ); + if (result.ok != 1) throw newError(1000, result); + this.set(result.value); + } + + const changeStreamId = uuid(); + const changeStreamData = { + _id: changeStreamId, + operation: StreamOperationType.update, + collection: this.collectionName, + documentKey: this._id, + updatedValues: dataThatHaveBeenChanged, + beforeUpdateValues: dataBeforeChange, + hookStatus: PostHookStatus.new, + createdAt: new Date(), + updateAt: new Date(), + }; + await (await DB.getInstance()).db.collection('stream.changes').insertOne(changeStreamData); + + if (this.afterUpdate && (await asyncHook(changeStreamData))) { + await this.afterUpdate(changeStreamData); + + await (await DB.getInstance()).db + .collection('stream.changes') + .updateOne({ _id: changeStreamId }, { $set: { hookStatus: PostHookStatus.completed } }); + } + + return this; + } catch (error) { + throw error; + } + } + + public async updateOneCustom(query: FilterQuery, set: UpdateQuery, ctx: ApolloContext | null): Promise { + try { + let dataBeforeChange: any; + + if (ctx) { + const account = ctx.ctx.user; + // Check if doc exists + if (!this._id) await this.getOne(query, ctx); + dataBeforeChange = this.get(); + + checkPermissions(this, account, 'w'); + + // Validate new data that will go into the DB + // const validation = await this.validate(); + // if (!validation.success) return validation; + + // Prepare data + this.updatedAt = new Date(); + + const result = await (await this.db()).findOneAndUpdate(query, set, { upsert: false, returnOriginal: false }); + if (result.ok != 1) throw newError(1000, result); + + if (result.value) this.set(result.value); + } else { + if (!this._id) await this.getOne(query, null); + dataBeforeChange = this.get(); + + // Prepare data + this.updatedAt = new Date(); + + const result = await (await this.db()).findOneAndUpdate(query, set, { upsert: false, returnOriginal: false }); + if (result.ok != 1) throw newError(1000, result); + + if (result.value) this.set(result.value); + } + + const changeStreamId = uuid(); + const changeStreamData = { + _id: changeStreamId, + operation: StreamOperationType.update, + collection: this.collectionName, + documentKey: this._id, + hookStatus: PostHookStatus.new, + updatedValues: JSON.stringify(set), + beforeUpdateValues: dataBeforeChange, + createdAt: new Date(), + updateAt: new Date(), + }; + await (await DB.getInstance()).db.collection('stream.changes').insertOne(changeStreamData); + if (this.afterUpdate && (await asyncHook(changeStreamData))) { + await this.afterUpdate(changeStreamData); + + await (await DB.getInstance()).db + .collection('stream.changes') + .updateOne({ _id: changeStreamId }, { $set: { hookStatus: PostHookStatus.completed } }); + } + + return this; + } catch (error) { + throw error; + } + } + + public async deleteOne(query: FilterQuery, ctx: ApolloContext | null): Promise { + try { + if (ctx) { + const account = ctx.ctx.user; + if (!this._id) await this.getOne(query, ctx); + checkPermissions(this, account, 'd'); + + if (this.beforeDelete) await this.beforeDelete(ctx); + + const result = await (await this.db()).findOneAndDelete(query); + + if (result.ok != 1) throw newError(1001); + + const deletedModel = '_deleted_' + this.collectionName; + await (await DB.getInstance()).db.collection(deletedModel).insertOne({ ...result.value, deletedBy: account._id }); + } else { + if (!this._id) await this.getOne(query, null); + + if (this.beforeDelete) await this.beforeDelete(); + + const result = await (await this.db()).findOneAndDelete(query); + + if (result.ok != 1) throw newError(1001); + + const deletedModel = '_deleted_' + this.collectionName; + await (await DB.getInstance()).db.collection(deletedModel).insertOne(result.value); + } + const changeStreamId = uuid(); + + const changeStreamData = { + _id: changeStreamId, + operation: StreamOperationType.delete, + collection: this.collectionName, + documentKey: this._id, + hookStatus: PostHookStatus.new, + beforeUpdateValues: this.get(), + createdAt: new Date(), + updateAt: new Date(), + }; + await (await DB.getInstance()).db.collection('stream.changes').insertOne(changeStreamData); + + if (this.afterDelete && (await asyncHook(changeStreamData))) { + await this.afterDelete(changeStreamData); + + await (await DB.getInstance()).db + .collection('stream.changes') + .updateOne({ _id: changeStreamId }, { $set: { hookStatus: PostHookStatus.completed } }); + } + + return this; + } catch (error) { + throw error; + } + } + + /* + * + * Permissions + * + */ + + public async updateQueryWithPermission(query: FilterQuery, ctx: ApolloContext): Promise> { + const account = ctx.ctx.user; + const organisationId = ctx.ctx.organisationId; + + // We add the permission (on r field) to make sure that kind of user has access to the ressource + const finalQuery = addPermissionToQuery(account, 'get', { ...query }); + + // We check if it comes from an organisation + // Check of the user's correct authorisation on that organisation comes from the middleware + if (organisationId) { + finalQuery['organisationId'] = { $eq: organisationId }; + } else { + finalQuery['organisationId'] = { $exists: false }; + } + + return finalQuery; + } + + // public async updateOnePermission( + // query: FilterQuery, + // add: 'add' | 'remove', + // type: 'r' | 'w' | 'd', + // newPerm: string, + // account?: AccountModel, + // ): Promise { + // // Get the object from the DB + // try { + // // Check if doc exists + // // TODO : if (this.get() returns, then don't) + // await this.getOne(query, account); + // if (account) this.checkPermissions(account.get(), 'w'); + + // // Validate new data that will go into the DB + // // const validation = await this.validate(); + // // if (!validation.success) return validation; + + // // Prepare data + // this.updatedAt = new Date(); + + // if (add == 'add') { + // const result = await (await this.db()).findOneAndUpdate( + // query, + // { $addToSet: { [type]: newPerm } }, + // { upsert: false, returnOriginal: false }, + // ); + // if (result.ok != 1) throw newError('general.dbError', '400', result); + + // this.set(result.value); + // } else if (add == 'remove') { + // const result = await (await this.db()).findOneAndUpdate( + // query, + // { $pull: { [type]: newPerm } }, + // { upsert: false, returnOriginal: false }, + // ); + // if (result.ok != 1) throw newError('general.dbError', '400', result); + + // this.set(result.value); + // } + + // return this; + // } catch (error) { + // throw error; + // } + // } + + public async addOneSubModel( + parentQ: { _id: string }, + subModelRefField: string, + subModelFieldName: keyof this, + subModel: SUB, + subModelInput: any, + ctx: ApolloContext | null, + ): Promise { + try { + subModelInput[subModelRefField] = parentQ._id; + const subM = await subModel.saveOne(subModelInput, ctx); + await this.updateOneCustom(parentQ as any, { $push: { [`${subModelFieldName}`]: subM._id } } as any, null); + return subM; + } catch (error) { + throw error; + } + } + + public async editOneSubModel( + subModel: SUB, + subModelId: string, + subModelInput: any, + ctx: ApolloContext | null, + ): Promise { + try { + // No need to check if belongs to the correct organisation, the middleware does that + const subM = await subModel.updateOne({ _id: subModelId } as any, subModelInput, ctx); + return subM; + } catch (error) { + throw error; + } + } + + public async deleteOneSubModel( + parentQ: FilterQuery, + subModelFieldName: keyof this, + subModel: SUB, + subModelId: string, + ctx: ApolloContext | null, + ): Promise { + try { + const subM = await subModel.deleteOne({ _id: subModelId } as any, ctx); + await this.updateOneCustom(parentQ, { $pull: { [`${subModelFieldName}`]: subM._id } } as any, null); + return subM; + } catch (error) { + throw error; + } + } + + public getOneSubRessource(subRessoureFieldName: keyof this | string, subRessoureId: string): any | undefined { + try { + const subress = _.get(this, subRessoureFieldName); + if (subress) { + const ind = _.findIndex(subress, { _id: subRessoureId }); + if (ind !== -1) return subress[ind]; + } + return; + } catch (error) { + throw error; + } + } + + public getDefaultSubRessource(subRessoureFieldName: keyof this | string): any | undefined { + try { + const subress = _.get(this, subRessoureFieldName); + if (subress) { + const ind = _.findIndex(subress, { default: true }); + if (ind !== -1) return subress[ind]; + return subress[0]; + } + return; + } catch (error) { + throw error; + } + } + + public async addOneSubRessource( + parentQ: { _id: string }, + subRessoureFieldName: keyof this | string, + subRessoure: any, + ctx: ApolloContext | null, + ): Promise { + try { + if (subRessoure.default && _.get(this, subRessoureFieldName)) { + // Remove all default from others + await this.updateOneCustom( + { _id: this._id } as any, + { + $set: { [`${subRessoureFieldName}.$[].default`]: false } as any, + }, + ctx, + ); + } else subRessoure.default = false; + + // Generate ID if not in the model + if (!subRessoure._id) subRessoure._id = uuid(); + await this.updateOneCustom(parentQ as any, { $push: { [`${subRessoureFieldName}`]: subRessoure } } as any, ctx); + return subRessoure; + } catch (error) { + throw error; + } + } + + public async editOneSubRessource( + parentQ: { _id: string }, + subRessoureFieldName: keyof this | string, + subRessoureId: string, + subRessoure: any, + ctx: ApolloContext | null, + ): Promise { + try { + if (!this.getOneSubRessource(subRessoureFieldName, subRessoureId)) throw newError(404); + + if (subRessoure.default) { + // Remove all default from others + await this.updateOneCustom( + { _id: this._id } as any, + { + $set: { [`${subRessoureFieldName}.$[].default`]: false } as any, + }, + ctx, + ); + } else subRessoure.default = false; + + const filterQ = { + ...parentQ, + [`${subRessoureFieldName}._id`]: subRessoureId, + }; + await this.updateOneCustom( + filterQ as any, + { + $set: { + [`${subRessoureFieldName}.$`]: { + ...subRessoure, + _id: subRessoureId, + }, + } as any, + }, + ctx, + ); + return { + ...subRessoure, + _id: subRessoureId, + }; + } catch (error) { + throw error; + } + } + + public async deleteOneSubRessource( + parentQ: { _id: string }, + subRessoureFieldName: keyof this | string, + subRessoureId: string, + ctx: ApolloContext | null, + ): Promise { + try { + if (!this.getOneSubRessource(subRessoureFieldName, subRessoureId)) throw newError(404); + + const filterQ = { + ...parentQ, + [`${subRessoureFieldName}._id`]: subRessoureId, + }; + + await this.updateOneCustom(filterQ as any, { $pull: { [`${subRessoureFieldName}`]: { _id: subRessoureId } } } as any, ctx); + return subRessoureId; + } catch (error) { + throw error; + } + } +} diff --git a/lib/seed/graphql/BaseService.ts b/lib/seed/graphql/BaseService.ts new file mode 100644 index 0000000..5ac8931 --- /dev/null +++ b/lib/seed/graphql/BaseService.ts @@ -0,0 +1,122 @@ +import _ from 'lodash'; + +import { ApolloError } from 'apollo-server-lambda'; + +import { GetArgs } from './Request'; +import { ApolloContext } from '@seed/interfaces/context'; +import { BaseGraphModel } from './BaseGraphModel'; +import { NewPermissionInput } from '@seed/interfaces/components'; +import { baseSearchFunction } from '@seed/services/database/DBRequestService'; +import { newError } from '@seed/helpers/Error'; + +export const getOneGeneric = async (model: T, id: string, ctx: ApolloContext): Promise => { + try { + await model.getOne({ _id: id } as any, ctx); + return model.get(); + } catch (error) { + throw error; + } +}; + +export const addOneGeneric = async (model: T, body: Partial, ctx: ApolloContext): Promise => { + try { + await model.saveOne(body, ctx); + return model.get(); + } catch (error) { + throw error; + } +}; + +export const editOneGeneric = async (model: T, id: string, body: Partial, ctx: ApolloContext): Promise => { + try { + await model.updateOne({ _id: id } as any, body, ctx); + return model.get(); + } catch (error) { + throw error; + } +}; + +export const deleteOneGeneric = async (model: T, id: string, ctx: ApolloContext): Promise => { + try { + await model.deleteOne({ _id: id } as any, ctx); + return model.get(); + } catch (error) { + throw error; + } +}; + +export const getManyGeneric = async (model: T, query: any, pagination: GetArgs, ctx: ApolloContext): Promise => { + return baseSearchFunction({ + model: model, + query: query, + pagination: pagination, + ctx: ctx, + }); +}; + +export const getCountGeneric = async (model: T, baseArguments: any, ctx: ApolloContext): Promise => { + // eslint-disable-next-line @typescript-eslint/no-unused-vars + const { pagination, ...args } = baseArguments; + + const result = await baseSearchFunction({ + model: model, + query: args, + count: true, + ctx: ctx, + }); + + return result.count || 0; +}; + +export const getManyGenericWithArgs = async (model: T, baseArguments: any, ctx: ApolloContext): Promise => { + try { + const { pagination, ...args } = baseArguments; + return baseSearchFunction({ + model: model, + query: args, + pagination: pagination, + ctx: ctx, + }); + } catch (error) { + throw error; + } +}; + +export const permissionsAddOne = async (model: any, id: string, input: NewPermissionInput, ctx: ApolloContext): Promise => { + try { + await model.updateOneCustom({ _id: id }, { $addToSet: { [`${input.permissionType}`]: input.permission } }, ctx); + return model.get(); + } catch (error) { + throw error; + } +}; + +export const permissionsRemoveOne = async (model: any, id: string, input: NewPermissionInput, ctx: ApolloContext): Promise => { + try { + await model.updateOneCustom({ _id: id }, { $pull: { [`${input.permissionType}`]: input.permission } }, ctx); + return model.get(); + } catch (error) { + throw error; + } +}; + +export const publishOneGeneric = async (model: T, id: string, ctx: ApolloContext): Promise => { + try { + const missingInputs: string[] = []; + // TODO THE CHECK OF THE INPUTS + + if (missingInputs.length > 0) throw newError(3000, missingInputs); + + return editOneGeneric(model, id, { published: true, publicationDate: new Date() } as any, ctx); + } catch (error) { + throw error; + } +}; + +export const unpublishOneGeneric = async (model: T, id: string, ctx: ApolloContext): Promise => { + try { + return editOneGeneric(model, id, { published: false } as any, ctx); + } catch (error) { + throw error; + } +}; diff --git a/lib/seed/graphql/Middleware.ts b/lib/seed/graphql/Middleware.ts new file mode 100644 index 0000000..de4919e --- /dev/null +++ b/lib/seed/graphql/Middleware.ts @@ -0,0 +1,204 @@ +import _ from 'lodash'; +import { APIGatewayProxyEvent, Context } from 'aws-lambda'; +import { ApolloError } from 'apollo-server-lambda'; +import { AuthChecker, createMethodDecorator } from 'type-graphql'; + +import Firebase from '@seed/services/auth/FirebaseService'; +import { ApolloContext } from '@seed/interfaces/context'; + +import AccountModel from 'src/accounts/account.model'; +import Loaders from '@src/__indexes/__loaders'; +import { AccountTypeEnum } from '@src/accounts/account.components'; + +import { simpleEstimator, getComplexity, fieldExtensionsEstimator } from 'graphql-query-complexity'; +import { clog } from '@seed/helpers/Utils'; +import { newError } from '@seed/helpers/Error'; +import { SecurityLevelEnum } from '@services/module-accounts/account.components'; + +export const ctxMiddleware = async (ctx: { event?: APIGatewayProxyEvent; context?: Context; req?: any }): Promise => { + const user: AccountModel = new AccountModel(); + let headers: any = {}; + let organisationId = null; + + try { + if (ctx.event) headers = ctx.event.headers; + else if (ctx.req) headers = ctx.req.headers; + + /* + * Deal with Auth + */ + + const authorization: string = headers['x-authorization'] || headers.authorization || headers.Authorization; + + if (authorization) { + // Verify the token + const firebaseRes = await Firebase.getInstance().tokenIdentify(authorization); + // Verify that exists in DB + await user.getOne({ _id: firebaseRes.uid }, null); + + /* + * Deal with Security + */ + + const securityCheck = user.securityCheck(); + if (!securityCheck) { + throw newError(2400, { + security: user.security, + securityLevel: process.env.SECURITY_LEVEL || SecurityLevelEnum.nothing, + }); + } + + /* + * Deal with Organisation + */ + organisationId = headers.organisationId || headers.organisationid; + if (organisationId) { + // Check if admin + if (user.types && user.types.includes(AccountTypeEnum.admin)) { + console.log('[ADMIN BYPASS]'); + } else if (!user._id || !user.organisationIds || !user.organisationIds?.includes(organisationId)) + throw newError(2101, { + accountOrganisationIds: user.organisationIds, + organisationId: organisationId, + }); + + console.log('[ORG]', organisationId); + } + + console.log('[AUTH - ]', user.types, user?._id); + } else { + console.log('no auth'); + } + } catch (error) { + throw error; + } + + return { + event: ctx.event || ctx.req || {}, + context: ctx.context || ctx.req || {}, + ctx: { + organisationId: organisationId, + user: user, + loaders: new Loaders(), + }, + }; +}; + +export const authMiddleware: AuthChecker = ({ root, args, context, info }, roles) => { + const user = context.ctx.user; + + if (!user || !user._id) { + throw newError(2004); + } + + if (roles.length > 0) { + roles.push(AccountTypeEnum.admin); + + const intersection = _.intersection(roles, user.types); + if (intersection.length > 0) return true; + + // no roles matched, restrict access + throw newError(2000, { you: user.types, allowed: roles }); + } + + return true; +}; + +export const checkOrganisation = (check = false): any => { + return createMethodDecorator(async ({ root, args, context, info }, next) => { + if (check && !(context as any).ctx.organisationId) throw newError(2102, '400'); + return next(); + }); +}; + +export const totalPublic = (check = false): any => { + return createMethodDecorator(async ({ root, args, context, info }, next) => { + if (check) (context as any).ctx.noOrganisationCheck = true; + return next(); + }); +}; + +export const errorMiddleware = (err): any => { + // Don't give the specific errors to the client. + + if (err.extensions) { + const { translations, ...data } = err.extensions; + const errorToReturn = { + code: err.extensions.code, + message: err.extensions.message || err.message, + translations, + data: process.env.NODE_ENV == 'production' ? { message: 'see logs' } : data, + }; + clog(errorToReturn); + + return errorToReturn; + } + + // Otherwise return the original error. The error can also + // be manipulated in other ways, so long as it's returned. + return err; +}; + +export const complexityMiddleware = (schema: any, variables: any, query: any) => { + const maximumComplexity = process.env.GRAPH_COMPLEXITY ? parseInt(process.env.GRAPH_COMPLEXITY) : 1000; + + const complexity = getComplexity({ + schema: schema, + + // The maximum allowed query complexity, queries above this threshold will be rejected + + // The query variables. This is needed because the variables are not available + // in the visitor of the graphql-js library + variables: variables, + + // specify operation name only when pass multi-operation documents + query: query, + + // Add any number of estimators. The estimators are invoked in order, the first + // numeric value that is being returned by an estimator is used as the field complexity. + // If no estimator returns a value, an exception is raised. + estimators: [ + // Add more estimators here... + fieldExtensionsEstimator(), + // This will assign each field a complexity of 1 if no other estimator + // returned a value. + simpleEstimator({ + defaultComplexity: 1, + }), + ], + }); + + // Here we can react to the calculated complexity, + // like compare it with max and throw error when the threshold is reached. + if (complexity >= maximumComplexity) { + throw new Error(`Sorry, too complicated query! ${complexity} is over ${maximumComplexity} that is the max allowed complexity.`); + } + // And here we can e.g. subtract the complexity point from hourly API calls limit. + console.log('Used query complexity points:', complexity); +}; + +export const oneToManyComplexity = ({ args, childComplexity }): number => { + if (args.pagination && args.pagination.limit) return childComplexity * args.pagination.limit; + else return childComplexity * 10; +}; + +export const oneToManyComplexityNoLoader = ({ args, childComplexity }): number => { + if (args.pagination && args.pagination.limit) return childComplexity * args.pagination.limit * 100; + else return childComplexity * 10 * 100; +}; + +export const createApolloContext = async (accountId?: string, organisationId: string | null = null): Promise => { + const user: AccountModel = new AccountModel(); + if (accountId) { + await user.getOne({ _id: accountId }, null); + } + return { + event: {} as any, + context: {} as any, + ctx: { + organisationId, + user: user, + loaders: new Loaders(), + }, + }; +}; diff --git a/lib/seed/graphql/MiddlewareV2.ts b/lib/seed/graphql/MiddlewareV2.ts new file mode 100644 index 0000000..e0aae0e --- /dev/null +++ b/lib/seed/graphql/MiddlewareV2.ts @@ -0,0 +1,132 @@ +import _ from 'lodash'; +import { APIGatewayProxyEvent, Context } from 'aws-lambda'; +import { ApolloError } from 'apollo-server-lambda'; +import { ClassType, createMethodDecorator, MiddlewareFn } from 'type-graphql'; + +import Firebase from '@seed/services/auth/FirebaseService'; +import { ApolloContext } from '@seed/interfaces/context'; + +import AccountModel from 'src/accounts/account.model'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { validateOrReject } from 'class-validator'; +import { plainToClass } from 'class-transformer'; +import { newError } from '@seed/helpers/Error'; + +export interface EngineMiddlewareInput { + authorization?: AccountTypeEnum[]; + apiKey?: boolean; + checkOrganisation?: boolean; + noOrganisationCheck?: boolean; + noPermissionCheck?: boolean; + validations?: { + inputName?: string; + schema: ClassType; + }[]; +} + +export const checkApiKey = async ({ context }): Promise => { + const user: AccountModel = new AccountModel(); + let headers: any = {}; + if (context.event) headers = context.event.headers; + else if (context.req) headers = context.req.headers; + + /* + * Deal with Auth + */ + + const apiKey: string = headers['x-api-key']; + + if (apiKey) { + // Verify that exists in DB + try { + const account = await user.getOne({ 'apiKeys.key': apiKey }, null); + context.ctx.user = account; + return true; + } catch (error) { + throw newError(2200, '403'); + } + } else throw newError(2202, { required: 'X-API-Key' }); +}; + +export const checkAuth = ({ context }, user, roles: string[]): boolean => { + let headers: any = {}; + if (context.event) headers = context.event.headers; + else if (context.req) headers = context.req.headers; + + if (roles.includes(AccountTypeEnum.public)) return true; + + const authorization: string = headers['x-authorization'] || headers.authorization || headers.Authorization; + if (!authorization) { + throw newError(2002); + } + + if (!user || !user._id) { + throw newError(2004, '403'); + } + + if (roles.length > 0) { + roles.push(AccountTypeEnum.admin); + user.types.push(AccountTypeEnum.public); + + const rolesToVerif = _.uniq(roles); + const userTypes = _.uniq(user.types); + + const intersection = _.intersection(rolesToVerif, userTypes); + if (intersection.length > 0) return true; + + // no roles matched, restrict access + throw newError(2000, { you: user.types, allowed: roles }); + } + + return true; +}; + +export const EngineMiddleware = (init: EngineMiddlewareInput): any => { + return createMethodDecorator(async (data, next) => { + // console.log('engine', data); + const { context, args } = data; + let returnError; + + if (init.authorization) { + const user = (context as any).ctx.user; + try { + checkAuth({ context }, user, init.authorization); + } catch (error) { + returnError = error; + } + } + if (init.apiKey) { + try { + await checkApiKey({ context }); + returnError = null; + } catch (error) { + returnError = error; + } + } + + if (returnError) throw returnError; + + if (init.checkOrganisation && !(context as any).ctx.organisationId) throw newError(2102, '400'); + if (init.noOrganisationCheck) (context as any).ctx.noOrganisationCheck = true; + if (init.noPermissionCheck) (context as any).ctx.noPermissionCheck = true; + + if (init.validations) + for (let index = 0; index < init.validations.length; index++) { + const element = init.validations[index]; + if (!element.inputName) element.inputName = 'input'; + try { + const instance = plainToClass(element.schema, args[element.inputName]); + await validateOrReject(instance); + } catch (error) { + throw newError(3001, error); + } + } + + return next(); + }); +}; + +export const inputValidation: MiddlewareFn = async ({ root, args, context, info }, next) => { + console.log({ root, args, context, info }); + return next(); +}; diff --git a/lib/seed/graphql/Request.ts b/lib/seed/graphql/Request.ts new file mode 100644 index 0000000..f2a6bec --- /dev/null +++ b/lib/seed/graphql/Request.ts @@ -0,0 +1,54 @@ +import { Field, Int, InputType, ArgsType } from 'type-graphql'; +import { Min, Max } from 'class-validator'; +import { DateTabType, RangeType } from '@seed/interfaces/components'; +import { DateRangeComponent } from '@seed/interfaces/components.dates'; + +@InputType() +export class GetArgs { + @Field(() => Int) + @Min(1) + @Max(100) + limit: number; + + @Field(() => Int) + @Min(0) + skip: number; + + @Field({ nullable: true }) + sort?: string; +} + +@ArgsType() +export class GetManyArgs { + @Field(() => [String], { nullable: true }) + _ids?: string[]; + + @Field(() => String, { nullable: true }) + search?: string; + + @Field(() => Date, { nullable: true }) + afterCreatedAt?: Date; + @Field(() => Date, { nullable: true }) + afterUpdatedAt?: Date; + + @Field(() => GetArgs, { nullable: true }) + pagination?: GetArgs; +} + +@ArgsType() +export class GetManyArgsWithDateRange extends GetManyArgs { + @Field({ nullable: true }) + startDate?: Date; + + @Field({ nullable: true }) + endDate?: Date; + + @Field(() => RangeType, { nullable: true }) + rangeType?: RangeType; + + @Field(() => DateTabType, { nullable: true }) + dateTabType?: DateTabType; + + @Field(() => DateRangeComponent, { nullable: true }) + dateRange?: DateRangeComponent; +} diff --git a/lib/seed/graphql/Server.ts b/lib/seed/graphql/Server.ts new file mode 100644 index 0000000..a0a4946 --- /dev/null +++ b/lib/seed/graphql/Server.ts @@ -0,0 +1,102 @@ +import { nullToUndefined } from '@seed/helpers/Utils'; +import { ApolloServer } from 'apollo-server-lambda'; +import { buildSchema } from 'type-graphql'; +import { ctxMiddleware, authMiddleware, errorMiddleware, complexityMiddleware } from './Middleware'; + +import { SettingsCache } from './Settings'; + +export async function createServer(resolversEntry: any): Promise { + const schema = await buildSchema({ + resolvers: resolversEntry, + authChecker: authMiddleware, + validate: false, + }); + + const server = new ApolloServer({ + // typeDefs, + // resolvers, + schema, + formatError: errorMiddleware, + formatResponse: (response): any => { + return nullToUndefined(response); + }, + context: ctxMiddleware, + tracing: true, + plugins: [ + { + // eslint-disable-next-line @typescript-eslint/explicit-function-return-type + requestDidStart: () => ({ + // eslint-disable-next-line @typescript-eslint/explicit-function-return-type + didResolveOperation({ request, document }) { + return complexityMiddleware(schema, request.variables, document); + }, + }), + }, + ], + }); + return server; +} + +export class ServerInstance { + private static instance: ServerInstance; + public handler!: any; + + // eslint-disable-next-line @typescript-eslint/no-empty-function + private constructor() {} + + public static async getInstance(resolversEntry?: any): Promise { + const settingsI = SettingsCache.getInstance(); + await settingsI.refreshCache(); + + if (!this.instance) { + const instance = new ServerInstance(); + + try { + // let { typeDefs, resolvers } = await buildTypeDefsAndResolvers({ + // resolvers: resolversEntry, + // authChecker: authMiddleware, + // }); + + // typeDefs = typeDefs.replace('scalar Upload', 'scalar UploadFix'); + // resolvers = resolvers; + + const server = await createServer(resolversEntry); + instance.handler = server.createHandler({ + cors: { + origin: '*', + credentials: true, + allowedHeaders: 'Content-Type,Authorization,X-Api-Key,X-Amz-Security-Token,X-Amz-User-Agent,organisationid', + }, + }); + console.log(`[GraphQL] Using new server in ${process.env.NODE_ENV}`); + this.instance = instance; + } catch (error) { + console.log('[GRAPH ERROR]', error); + throw error; + } + } else { + console.log(`[GraphQL] Using same server in ${process.env.NODE_ENV}`); + } + + return this.instance; + } + + public run(event: AWSLambda.APIGatewayProxyEvent, context: AWSLambda.Context): Promise { + try { + console.log('BEGIN ACTION', event.body ? JSON.parse(event.body) : 'NO BODY'); + } catch (error) { + console.log('BEGIN ACTION - Not JSON'); + } + return new Promise((resolve, reject) => { + const callback = (error: any, body: unknown): void => { + console.log('END ACTION'); + if (error) { + console.error(error); + reject(error); + } else resolve(body); + }; + + this.handler(event, context, callback); + }); + } +} diff --git a/lib/seed/graphql/Settings.ts b/lib/seed/graphql/Settings.ts new file mode 100644 index 0000000..538de69 --- /dev/null +++ b/lib/seed/graphql/Settings.ts @@ -0,0 +1,202 @@ +/* eslint-disable @typescript-eslint/explicit-function-return-type */ +/* eslint-disable prettier/prettier */ +/* eslint-disable @typescript-eslint/no-use-before-define */ +import { DateTime } from 'luxon'; +import _ from 'lodash'; + +import env from '@config/.env.json'; +import packageFile from 'package.json'; + +import { SettingsType } from '@seed/interfaces/components'; + +import { ErrorsConfig, NotificationEnum } from '@config/config'; +import { EngineNotificationEnum } from '@seed/interfaces/components.notifications'; +import { TranslatableComponent } from '@src/__components/components'; +import { promiseAll } from '@seed/helpers/Utils'; +import DB from '@seed/services/database/DBService'; + +import { v4 as uuid } from 'uuid'; + +export interface GetSettingsOutput { + env: any[]; + errors: any[]; + emails: any[]; +} + +export interface EngineServerCacheInfo { + emailsContent: { + [key in NotificationEnum | EngineNotificationEnum]: any; + }; + errorsContent: { + [key: string]: { + message: string; + translations?: { + subject?: TranslatableComponent; + body?: TranslatableComponent; + }; + }; + }; +} + +export const getSettingsFromDB = async (): Promise => { + const settings = (await await (await DB.getInstance()).db.collection('settings').find().toArray()) as (any | any | any)[]; + + // filtering client side because not too much data and no need to index db + return { + env: _.filter(settings, { type: SettingsType.env }) as any[], + errors: _.filter(settings, { type: SettingsType.errors }) as any[], + emails: _.filter(settings, { type: SettingsType.emails }) as any[], + }; +}; + +export const syncEnvFromDB = (currentEnvInDB: GetSettingsOutput): void => { + // update env + currentEnvInDB.env.map((env) => { + process.env[env.key] = env.value; + }); +}; + +export const initEnvironement = async (env: any): Promise => { + const currentEnv = process.env.NODE_ENV && env[process.env.NODE_ENV] ? env[process.env.NODE_ENV] : env.default; + + for (const k in currentEnv) { + process.env[k] = currentEnv[k]; + } + + // Init also Package.json + process.env.awsRegion = packageFile.devops.awsRegion; + + const currentSettings = await getSettingsFromDB(); + syncEnvFromDB(currentSettings); + + // Force the local to be > DB + if (process.env.NODE_ENV == 'local') { + for (const k in currentEnv) { + process.env[k] = currentEnv[k]; + } + } + + console.log('[Server - Env] Loaded'); + + return currentSettings; +}; + +export class SettingsCache { + private static instance: SettingsCache; + private lastCacheDate: DateTime; + public cache: EngineServerCacheInfo; + + // eslint-disable-next-line @typescript-eslint/no-empty-function + private constructor() {} + + public static getInstance(): SettingsCache { + if (!this.instance) { + this.instance = new SettingsCache(); + } + + return this.instance; + } + + public async refreshCache(): Promise { + // check if needs to refresh the cache + if (!this.lastCacheDate || this.lastCacheDate.diff(DateTime.local(), 'minutes').minutes > 5) { + const settings = await initEnvironement(env); + + const promises: Promise[] = []; + + // Map for errors + const errorsContent = await mapErrorContent(settings, promises); + + // Map for emails + const emailsContent: { + [key in NotificationEnum | EngineNotificationEnum]: any; + } = await mapEmailSettings(settings, promises); + + this.cache = { + errorsContent, + emailsContent, + }; + this.lastCacheDate = DateTime.local(); + + if (promises.length > 0) { + console.log('Creating defaut in DB'); + await promiseAll(promises); + } + } + } +} +async function mapErrorContent(settings: GetSettingsOutput, newData: Promise[]) { + const errorsContent: { + [key: string]: { + message: string; + translations?: { + subject?: TranslatableComponent; + body?: TranslatableComponent; + }; + }; + } = {}; + for (const key in ErrorsConfig) { + if (Object.prototype.hasOwnProperty.call(ErrorsConfig, key)) { + const errIndex = _.findIndex(settings.errors, { key: key }); + errorsContent[key] = { + message: ErrorsConfig[key], + }; + if (errIndex === -1) { + console.error('[SETTINGS - ERROR] Creating default', key); + newData.push( + (await DB.getInstance()).db.collection('settings').insertOne({ + key, + type: SettingsType.errors, + value: ErrorsConfig[key], + body: new TranslatableComponent(), + _id: uuid(), + r: ['admin'], + w: ['admin'], + d: ['admin'], + createdAt: new Date(), + updatedAt: new Date(), + }), + ); + } else { + errorsContent[key].translations = { + subject: settings.errors[errIndex].subject, + body: settings.errors[errIndex].body, + }; + } + } + } + return errorsContent; +} + +async function mapEmailSettings(settings: GetSettingsOutput, newData: Promise[]) { + const emails = _.groupBy(settings.emails, 'key'); + const emailsContent: { + [key in NotificationEnum | EngineNotificationEnum]: any; + } = {} as any; + + const enumConcat = [...Object.keys(NotificationEnum), ...Object.keys(EngineNotificationEnum)]; + + enumConcat.forEach(async (element) => { + if (!emails[element]) { + console.error('[SETTINGS - EMAIL] Creating default', element); + newData.push( + (await DB.getInstance()).db.collection('settings').insertOne({ + key: element, + type: SettingsType.emails, + body: new TranslatableComponent(), + custom: false, + fromEmail: '', + fromName: '', + replyToEmail: '', + r: ['admin'], + w: ['admin'], + d: ['admin'], + _id: uuid(), + createdAt: new Date(), + updatedAt: new Date(), + }), + ); + } else emailsContent[element] = emails[element]; + }); + return emailsContent; +} diff --git a/lib/seed/graphql/baseModels/BaseContentModel.ts b/lib/seed/graphql/baseModels/BaseContentModel.ts new file mode 100644 index 0000000..f53e273 --- /dev/null +++ b/lib/seed/graphql/baseModels/BaseContentModel.ts @@ -0,0 +1,106 @@ +import { ObjectType, Field, ID, ArgsType, InputType, Int } from 'type-graphql'; + +import { BaseGraphModel } from '@seed/graphql/BaseGraphModel'; +import ImageComponent, { SEOField } from '@seed/interfaces/components'; +import { GetArgs, GetManyArgs } from '@seed/graphql/Request'; + +import { TranslatableComponent } from '@src/__components/components'; + +import { Permission } from '@seed/interfaces/permission'; +import BaseSEOModel, { NewBaseSEOInput, EditBaseSEOInput } from './BaseSeoModel'; +import CategoryModel from '@services/module-cms/functions/categories/category.model'; +import { AccountTypeEnum } from '@src/accounts/account.components'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.public], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class BaseContentModel extends BaseSEOModel { + public constructor(collectionName?: string, perm?: Permission) { + const cName = collectionName ? collectionName : 'articles'; + const permission = perm ? perm : permissions; + super(cName, permission); + } + + @Field() + published?: boolean; + @Field({ nullable: true }) + publicationDate?: Date; + + @Field({ nullable: true }) + private?: boolean; + + @Field(() => [String], { nullable: true }) + categoryIds?: string[]; + @Field(() => [CategoryModel], { nullable: 'itemsAndList' }) + getCategories?: CategoryModel[]; + + @Field(() => [ImageComponent], { nullable: true }) + downloadableRessources?: ImageComponent[]; + + @Field(() => TranslatableComponent, { nullable: true }) + testimonials?: TranslatableComponent; + + searchOptions(): string[] { + return []; + } +} + +@ArgsType() +export class BaseContentArgs extends GetManyArgs {} + +@InputType() +export class NewBaseContentInput extends NewBaseSEOInput implements Partial { + @Field(() => TranslatableComponent, { nullable: true }) + title?: TranslatableComponent; + @Field(() => TranslatableComponent, { nullable: true }) + teaser?: TranslatableComponent; + + @Field(() => ImageComponent, { nullable: true }) + cover?: ImageComponent; + @Field(() => ImageComponent, { nullable: true }) + thumbnail?: ImageComponent; + + @Field(() => [String], { nullable: true }) + categoryIds?: string[]; + @Field(() => [ImageComponent], { nullable: true }) + downloadableRessources?: ImageComponent[]; + + published = false; + + @Field({ nullable: true }) + private?: boolean; + + @Field(() => TranslatableComponent, { nullable: true }) + testimonials?: TranslatableComponent; +} + +@InputType() +export class EditBaseContentInput extends EditBaseSEOInput implements Partial { + @Field(() => TranslatableComponent, { nullable: true }) + title?: TranslatableComponent; + @Field(() => TranslatableComponent, { nullable: true }) + teaser?: TranslatableComponent; + @Field(() => TranslatableComponent, { nullable: true }) + content?: TranslatableComponent; + + @Field(() => ImageComponent, { nullable: true }) + cover?: ImageComponent; + @Field(() => ImageComponent, { nullable: true }) + thumbnail?: ImageComponent; + + @Field(() => [String], { nullable: true }) + categoryIds?: string[]; + @Field(() => [ImageComponent], { nullable: true }) + downloadableRessources?: ImageComponent[]; + + @Field({ nullable: true }) + private?: boolean; + + @Field(() => TranslatableComponent, { nullable: true }) + testimonials?: TranslatableComponent; +} diff --git a/lib/seed/graphql/baseModels/BaseSeoModel.ts b/lib/seed/graphql/baseModels/BaseSeoModel.ts new file mode 100644 index 0000000..3e6ea67 --- /dev/null +++ b/lib/seed/graphql/baseModels/BaseSeoModel.ts @@ -0,0 +1,103 @@ +import { ObjectType, Field, ID, ArgsType, InputType, Int } from 'type-graphql'; + +import { BaseGraphModel } from '@seed/graphql/BaseGraphModel'; +import ImageComponent, { SEOField } from '@seed/interfaces/components'; +import { GetArgs, GetManyArgs } from '@seed/graphql/Request'; + +import { TranslatableComponent } from '@src/__components/components'; + +import { Permission } from '@seed/interfaces/permission'; +import { AccountTypeEnum } from '@src/accounts/account.components'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.public], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class BaseSEOModel extends BaseGraphModel { + public constructor(collectionName?: string, perm?: Permission) { + const cName = collectionName ? collectionName : 'articles'; + const permission = perm ? perm : permissions; + super({ + // ...init, + collectionName: cName, + permissions: permission, + }); + } + + @Field(() => TranslatableComponent, { nullable: true }) + title?: TranslatableComponent; + @Field(() => TranslatableComponent, { nullable: true }) + teaser?: TranslatableComponent; + + @Field(() => ImageComponent, { nullable: true }) + cover?: ImageComponent; + @Field(() => ImageComponent, { nullable: true }) + thumbnail?: ImageComponent; + + @Field(() => [ImageComponent], { nullable: true }) + extraImages?: ImageComponent[]; + + @Field(() => TranslatableComponent, { nullable: true }) + content?: TranslatableComponent; + + @Field(() => SEOField, { nullable: true }) + seo?: SEOField; + @Field({ nullable: true }) + urls?: TranslatableComponent; + + searchOptions(): string[] { + return []; + } + filterOptions(): string[] { + return []; + } +} + +@ArgsType() +export class BaseSEOArgs extends GetManyArgs {} + +@InputType() +export class NewBaseSEOInput implements Partial { + @Field(() => TranslatableComponent, { nullable: true }) + title?: TranslatableComponent; + @Field(() => TranslatableComponent, { nullable: true }) + teaser?: TranslatableComponent; + + @Field(() => ImageComponent, { nullable: true }) + cover?: ImageComponent; + @Field(() => ImageComponent, { nullable: true }) + thumbnail?: ImageComponent; + + @Field(() => TranslatableComponent, { nullable: true }) + content?: TranslatableComponent; + + @Field(() => SEOField, { nullable: true }) + seo?: SEOField; + @Field({ nullable: true }) + urls?: TranslatableComponent; +} + +@InputType() +export class EditBaseSEOInput implements Partial { + @Field(() => TranslatableComponent, { nullable: true }) + title?: TranslatableComponent; + @Field(() => TranslatableComponent, { nullable: true }) + teaser?: TranslatableComponent; + + @Field(() => ImageComponent, { nullable: true }) + cover?: ImageComponent; + @Field(() => ImageComponent, { nullable: true }) + thumbnail?: ImageComponent; + + @Field(() => TranslatableComponent, { nullable: true }) + content?: TranslatableComponent; + + @Field(() => SEOField, { nullable: true }) + seo?: SEOField; + @Field({ nullable: true }) + urls?: TranslatableComponent; +} diff --git a/lib/seed/graphql/baseModels/BaseSeoSimpleModel.ts b/lib/seed/graphql/baseModels/BaseSeoSimpleModel.ts new file mode 100644 index 0000000..c39253b --- /dev/null +++ b/lib/seed/graphql/baseModels/BaseSeoSimpleModel.ts @@ -0,0 +1,107 @@ +import { ObjectType, Field, ID, ArgsType, InputType, Int } from 'type-graphql'; + +import { BaseGraphModel } from '@seed/graphql/BaseGraphModel'; +import ImageComponent, { SEOSimpleField } from '@seed/interfaces/components'; +import { GetArgs, GetManyArgs } from '@seed/graphql/Request'; + +import { Permission } from '@seed/interfaces/permission'; +import { AccountTypeEnum } from '@src/accounts/account.components'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.public], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class BaseSEOSimpleModel extends BaseGraphModel { + public constructor(collectionName?: string, perm?: Permission) { + const cName = collectionName ? collectionName : 'articles'; + const permission = perm ? perm : permissions; + super({ + // ...init, + collectionName: cName, + permissions: permission, + }); + } + + @Field({ nullable: true }) + title?: string; + @Field({ nullable: true }) + teaser?: string; + + @Field(() => ImageComponent, { nullable: true }) + cover?: ImageComponent; + @Field(() => ImageComponent, { nullable: true }) + thumbnail?: ImageComponent; + + @Field(() => [ImageComponent], { nullable: true }) + extraImages?: ImageComponent[]; + + @Field({ nullable: true }) + content?: string; + + @Field(() => SEOSimpleField, { nullable: true }) + seo?: SEOSimpleField; + @Field({ nullable: true }) + urls?: string; + + searchOptions(): string[] { + return []; + } + filterOptions(): string[] { + return []; + } +} + +@ArgsType() +export class BaseSEOArgs extends GetManyArgs {} + +@InputType() +export class NewBaseSEOSimpleInput implements Partial { + @Field({ nullable: true }) + title?: string; + @Field({ nullable: true }) + teaser?: string; + + @Field(() => ImageComponent, { nullable: true }) + cover?: ImageComponent; + @Field(() => ImageComponent, { nullable: true }) + thumbnail?: ImageComponent; + + @Field(() => [ImageComponent], { nullable: true }) + extraImages?: ImageComponent[]; + + @Field({ nullable: true }) + content?: string; + + @Field(() => SEOSimpleField, { nullable: true }) + seo?: SEOSimpleField; + @Field({ nullable: true }) + urls?: string; +} + +@InputType() +export class EditBaseSEOSimpleInput implements Partial { + @Field({ nullable: true }) + title?: string; + @Field({ nullable: true }) + teaser?: string; + + @Field(() => ImageComponent, { nullable: true }) + cover?: ImageComponent; + @Field(() => ImageComponent, { nullable: true }) + thumbnail?: ImageComponent; + + @Field(() => [ImageComponent], { nullable: true }) + extraImages?: ImageComponent[]; + + @Field({ nullable: true }) + content?: string; + + @Field(() => SEOSimpleField, { nullable: true }) + seo?: SEOSimpleField; + @Field({ nullable: true }) + urls?: string; +} diff --git a/lib/seed/graphql/baseResolvers/BasePublicResolver.ts b/lib/seed/graphql/baseResolvers/BasePublicResolver.ts new file mode 100644 index 0000000..60f133b --- /dev/null +++ b/lib/seed/graphql/baseResolvers/BasePublicResolver.ts @@ -0,0 +1,143 @@ +import { Query, Arg, Int, Resolver, Authorized, Ctx, Args, Mutation, ClassType } from 'type-graphql'; +import { ApolloContext } from '@seed/interfaces/context'; +import { getOneGeneric, getManyGenericWithArgs, addOneGeneric, editOneGeneric, deleteOneGeneric, getCountGeneric } from '../BaseService'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { totalPublic } from '../Middleware'; + +export const createBasePublicResolver = , ARGS extends ClassType>( + domain: string, + objectTypeCls: T, + argsType: ARGS, + pub = false, +): any => { + @Resolver({ isAbstract: true }) + abstract class BasePublicResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ + + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + @Query(() => objectTypeCls, { name: `${domain}GetOne` }) + @totalPublic(pub) + async getOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = new objectTypeCls(); + return getOneGeneric(model, id, ctx); + } + + @Query(() => [objectTypeCls], { name: `${domain}GetMany` }) + @totalPublic(pub) + async getMany(@Args((type) => argsType) args: ARGS, @Ctx() ctx: ApolloContext): Promise { + const model = new objectTypeCls(); + return getManyGenericWithArgs(model, args, ctx); + } + + @Query(() => Number, { name: `${domain}GetCount` }) + @totalPublic(pub) + async getCount(@Args((type) => argsType) args: ARGS, @Ctx() ctx: ApolloContext): Promise { + const model = new objectTypeCls(); + return getCountGeneric(model, args, ctx); + } + } + + return BasePublicResolver; +}; + +export const createBaseMyResolver = , ARGS extends ClassType>( + domain: string, + objectTypeCls: T, + argsType: ARGS, +): any => { + @Resolver({ isAbstract: true }) + abstract class BaseMyResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ + + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + @Query(() => objectTypeCls, { name: `${domain}GetOneMine` }) + @Authorized() + async getMineOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = new objectTypeCls(); + return getOneGeneric(model, id, ctx); + } + + @Query(() => [objectTypeCls], { name: `${domain}GetManyMine` }) + @Authorized() + async getMine(@Args((type) => argsType) args: ARGS, @Ctx() ctx: ApolloContext): Promise { + const model = new objectTypeCls(); + return getManyGenericWithArgs(model, args, ctx); + } + } + + return BaseMyResolver; +}; + +export const createBaseReadOnlyResolver = , ARGS extends ClassType>( + domain: string, + objectTypeCls: T, + argsType: ARGS, + auth: AccountTypeEnum[], +): any => { + @Resolver({ isAbstract: true }) + abstract class BaseReadOnlyResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ + + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + @Query(() => objectTypeCls, { name: `${domain}GetOne` }) + @Authorized(auth) + async getOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = new objectTypeCls(); + return getOneGeneric(model, id, ctx); + } + + @Query(() => [objectTypeCls], { name: `${domain}GetMany` }) + @Authorized(auth) + async getMany(@Args((type) => argsType) args: ARGS, @Ctx() ctx: ApolloContext): Promise { + const model = new objectTypeCls(); + return getManyGenericWithArgs(model, args, ctx); + } + } + + return BaseReadOnlyResolver; +}; diff --git a/lib/seed/graphql/baseResolvers/BaseResolver.ts b/lib/seed/graphql/baseResolvers/BaseResolver.ts new file mode 100644 index 0000000..66dc46d --- /dev/null +++ b/lib/seed/graphql/baseResolvers/BaseResolver.ts @@ -0,0 +1,100 @@ +import { Query, Arg, Int, Resolver, Authorized, Ctx, Args, Mutation, ClassType } from 'type-graphql'; +import { ApolloContext } from '@seed/interfaces/context'; +import { getOneGeneric, getManyGenericWithArgs, addOneGeneric, editOneGeneric, deleteOneGeneric, getCountGeneric } from '../BaseService'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { checkOrganisation as CheckOrganisation } from '../Middleware'; + +export const createBaseResolver = , ARGS extends ClassType, NEW extends ClassType, EDIT extends ClassType>( + domain: string, + objectTypeCls: T, + argsType: ARGS, + newInput: NEW, + editInput: EDIT, + auth: AccountTypeEnum[], + organisationCheck = false, +): any => { + @Resolver({ isAbstract: true }) + abstract class BaseResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ + + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + @Query(() => objectTypeCls, { name: `${domain}GetOne` }) + @Authorized(auth) + @CheckOrganisation(organisationCheck) + async getOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = new objectTypeCls(); + return getOneGeneric(model, id, ctx); + } + + @Query(() => [objectTypeCls], { name: `${domain}GetMany` }) + @Authorized(auth) + @CheckOrganisation(organisationCheck) + async getMany(@Args((type) => argsType) args: ARGS, @Ctx() ctx: ApolloContext): Promise { + const model = new objectTypeCls(); + return getManyGenericWithArgs(model, args, ctx); + } + + @Query(() => Number, { name: `${domain}GetCount` }) + @Authorized(auth) + @CheckOrganisation(organisationCheck) + async getCount(@Args((type) => argsType) args: ARGS, @Ctx() ctx: ApolloContext): Promise { + const model = new objectTypeCls(); + return getCountGeneric(model, args, ctx); + } + + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + @Mutation(() => objectTypeCls, { name: `${domain}AddOne` }) + @Authorized(auth) + @CheckOrganisation(organisationCheck) + async addOne(@Arg('input', (type) => newInput) input: NEW, @Ctx() ctx: ApolloContext): Promise { + const model = new objectTypeCls(); + if (model.onCreate) await model.onCreate(input); + + return addOneGeneric(model, input, ctx); + } + + @Mutation(() => objectTypeCls, { name: `${domain}EditOne` }) + @Authorized(auth) + @CheckOrganisation(organisationCheck) + async editOne(@Arg('id') id: string, @Arg('input', (type) => editInput) input: EDIT, @Ctx() ctx: ApolloContext): Promise { + const model = new objectTypeCls(); + if (model.onUpdate) await model.onUpdate(input); + + return editOneGeneric(model, id, input, ctx); + } + + @Mutation(() => objectTypeCls, { name: `${domain}DeleteOne` }) + @Authorized(auth) + @CheckOrganisation(organisationCheck) + async deleteOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = new objectTypeCls(); + if (model.onDelete) await model.onDelete(); + + return deleteOneGeneric(model, id, ctx); + } + } + + return BaseResolver; +}; diff --git a/lib/seed/graphql/genericResolvers/BaseGenericResolver.ts b/lib/seed/graphql/genericResolvers/BaseGenericResolver.ts new file mode 100644 index 0000000..3db683e --- /dev/null +++ b/lib/seed/graphql/genericResolvers/BaseGenericResolver.ts @@ -0,0 +1,97 @@ +import { Query, Arg, Resolver, Ctx, Args, Mutation, ClassType } from 'type-graphql'; +import { ApolloContext } from '@seed/interfaces/context'; +import { getOneGeneric, getManyGenericWithArgs, addOneGeneric, editOneGeneric, deleteOneGeneric, getCountGeneric } from '../BaseService'; +import { EngineMiddleware, EngineMiddlewareInput } from '../MiddlewareV2'; + +export const createGenericQueryResolver = , ARGS extends ClassType>(init: { + domain: string; + modelName: T; + argsType: ARGS; + customMiddlewareInput: EngineMiddlewareInput; +}): any => { + @Resolver({ isAbstract: true }) + abstract class BaseQueryResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ + + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + @Query(() => init.modelName, { name: `${init.domain}GetOne` }) + @EngineMiddleware(init.customMiddlewareInput) + async getOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = new init.modelName(); + return getOneGeneric(model, id, ctx); + } + + @Query(() => [init.modelName], { name: `${init.domain}GetMany` }) + @EngineMiddleware(init.customMiddlewareInput) + async getMany(@Args(() => init.argsType) args: ARGS, @Ctx() ctx: ApolloContext): Promise { + const model = new init.modelName(); + return getManyGenericWithArgs(model, args, ctx); + } + + @Query(() => Number, { name: `${init.domain}GetCount` }) + @EngineMiddleware(init.customMiddlewareInput) + async getCount(@Args(() => init.argsType) args: ARGS, @Ctx() ctx: ApolloContext): Promise { + const model = new init.modelName(); + return getCountGeneric(model, args, ctx); + } + } + + return BaseQueryResolver; +}; + +export const createGenericMutationResolver = , NEW extends ClassType, EDIT extends ClassType>(init: { + domain: string; + modelName: T; + newInput: NEW; + editInput: EDIT; + customMiddlewareInput: EngineMiddlewareInput; +}): any => { + @Resolver({ isAbstract: true }) + abstract class BaseMutationResolver { + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + @Mutation(() => init.modelName, { name: `${init.domain}AddOne` }) + @EngineMiddleware(init.customMiddlewareInput) + async addOne(@Arg('input', () => init.newInput) input: NEW, @Ctx() ctx: ApolloContext): Promise { + const model = new init.modelName(); + return addOneGeneric(model, input, ctx); + } + + @Mutation(() => init.modelName, { name: `${init.domain}EditOne` }) + @EngineMiddleware(init.customMiddlewareInput) + async editOne(@Arg('id') id: string, @Arg('input', () => init.editInput) input: EDIT, @Ctx() ctx: ApolloContext): Promise { + const model = new init.modelName(); + return editOneGeneric(model, id, input, ctx); + } + + @Mutation(() => init.modelName, { name: `${init.domain}DeleteOne` }) + @EngineMiddleware(init.customMiddlewareInput) + async deleteOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = new init.modelName(); + return deleteOneGeneric(model, id, ctx); + } + } + + return BaseMutationResolver; +}; diff --git a/lib/seed/graphql/genericResolvers/BaseResolver.ts b/lib/seed/graphql/genericResolvers/BaseResolver.ts new file mode 100644 index 0000000..66dc46d --- /dev/null +++ b/lib/seed/graphql/genericResolvers/BaseResolver.ts @@ -0,0 +1,100 @@ +import { Query, Arg, Int, Resolver, Authorized, Ctx, Args, Mutation, ClassType } from 'type-graphql'; +import { ApolloContext } from '@seed/interfaces/context'; +import { getOneGeneric, getManyGenericWithArgs, addOneGeneric, editOneGeneric, deleteOneGeneric, getCountGeneric } from '../BaseService'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { checkOrganisation as CheckOrganisation } from '../Middleware'; + +export const createBaseResolver = , ARGS extends ClassType, NEW extends ClassType, EDIT extends ClassType>( + domain: string, + objectTypeCls: T, + argsType: ARGS, + newInput: NEW, + editInput: EDIT, + auth: AccountTypeEnum[], + organisationCheck = false, +): any => { + @Resolver({ isAbstract: true }) + abstract class BaseResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ + + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + @Query(() => objectTypeCls, { name: `${domain}GetOne` }) + @Authorized(auth) + @CheckOrganisation(organisationCheck) + async getOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = new objectTypeCls(); + return getOneGeneric(model, id, ctx); + } + + @Query(() => [objectTypeCls], { name: `${domain}GetMany` }) + @Authorized(auth) + @CheckOrganisation(organisationCheck) + async getMany(@Args((type) => argsType) args: ARGS, @Ctx() ctx: ApolloContext): Promise { + const model = new objectTypeCls(); + return getManyGenericWithArgs(model, args, ctx); + } + + @Query(() => Number, { name: `${domain}GetCount` }) + @Authorized(auth) + @CheckOrganisation(organisationCheck) + async getCount(@Args((type) => argsType) args: ARGS, @Ctx() ctx: ApolloContext): Promise { + const model = new objectTypeCls(); + return getCountGeneric(model, args, ctx); + } + + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + @Mutation(() => objectTypeCls, { name: `${domain}AddOne` }) + @Authorized(auth) + @CheckOrganisation(organisationCheck) + async addOne(@Arg('input', (type) => newInput) input: NEW, @Ctx() ctx: ApolloContext): Promise { + const model = new objectTypeCls(); + if (model.onCreate) await model.onCreate(input); + + return addOneGeneric(model, input, ctx); + } + + @Mutation(() => objectTypeCls, { name: `${domain}EditOne` }) + @Authorized(auth) + @CheckOrganisation(organisationCheck) + async editOne(@Arg('id') id: string, @Arg('input', (type) => editInput) input: EDIT, @Ctx() ctx: ApolloContext): Promise { + const model = new objectTypeCls(); + if (model.onUpdate) await model.onUpdate(input); + + return editOneGeneric(model, id, input, ctx); + } + + @Mutation(() => objectTypeCls, { name: `${domain}DeleteOne` }) + @Authorized(auth) + @CheckOrganisation(organisationCheck) + async deleteOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = new objectTypeCls(); + if (model.onDelete) await model.onDelete(); + + return deleteOneGeneric(model, id, ctx); + } + } + + return BaseResolver; +}; diff --git a/lib/seed/helpers/Context.ts b/lib/seed/helpers/Context.ts new file mode 100644 index 0000000..925111b --- /dev/null +++ b/lib/seed/helpers/Context.ts @@ -0,0 +1,12 @@ +import { ApolloContextLoadersOnly } from '@seed/interfaces/context'; +import Loaders from '@src/__indexes/__loaders'; + +export const createContext = async (): Promise => { + return { + event: {} as any, + context: {} as any, + ctx: { + loaders: new Loaders(), + }, + }; +}; diff --git a/lib/seed/helpers/Error.ts b/lib/seed/helpers/Error.ts new file mode 100644 index 0000000..44bebdc --- /dev/null +++ b/lib/seed/helpers/Error.ts @@ -0,0 +1,43 @@ +import { ErrorsConfig } from '@config/config'; + +import { SettingsCache } from '@seed/graphql/Settings'; +import { ApolloError } from 'apollo-server-lambda'; +import { UtilStringHelper } from './Utils.string'; + +type ErrorKeys = keyof typeof ErrorsConfig; + +const utilStringHelper = new UtilStringHelper(); + +export const newError = (code: ErrorKeys, data?: any): ApolloError => { + const errorConfig = SettingsCache.getInstance().cache.errorsContent; + + const message = errorConfig[code].message; + const settings = errorConfig[code].translations; + + const translations = { + subject: {}, + body: {}, + }; + + if (settings) { + if (settings.body) { + for (const key in settings.body) { + if (Object.prototype.hasOwnProperty.call(settings.body, key)) { + translations.body[key] = utilStringHelper.interpolate(settings.body[key], data); + } + } + } + if (settings.subject) { + for (const key in settings.subject) { + if (Object.prototype.hasOwnProperty.call(settings.subject, key)) { + translations.subject[key] = utilStringHelper.interpolate(settings.subject[key], data); + } + } + } + } + + return new ApolloError(utilStringHelper.interpolate(message, data), code.toString(), { + translations, + data, + }); +}; diff --git a/lib/seed/helpers/FetchHelper.ts b/lib/seed/helpers/FetchHelper.ts new file mode 100644 index 0000000..5c146d1 --- /dev/null +++ b/lib/seed/helpers/FetchHelper.ts @@ -0,0 +1,64 @@ +import fetch from 'node-fetch'; +import querystring from 'querystring'; + +// convert object to a query string +export const apiCall = async (url: string, method: 'POST' | 'PUT' | 'PATCH' | 'GET' | 'DELETE', headers?: any, body?: any): Promise => { + const q: any = { + method: method, + headers: { + 'Content-Type': 'application/json', + }, + }; + + if (headers) q.headers = headers; + if (body) { + if (typeof body === 'string') q.body = body; + else q.body = JSON.stringify(body); + } + + const response = await fetch(url, q); + + const rs = await response.json(); + if (!response.ok) throw { error: rs }; + + return rs; +}; + +export const fetchCall = async (init: { + url: string; + method: 'POST' | 'PUT' | 'PATCH' | 'GET' | 'DELETE'; + query?: any; + headers?: any; + body?: any; + raw?: boolean; +}): Promise => { + let { url } = init; + const { method, query, headers, body, raw } = init; + const q: any = { + method: method, + headers: { + 'Content-Type': 'application/json', + }, + }; + + if (query) url = url + '?' + querystring.stringify(query, undefined, undefined, { encodeURIComponent: encodeURIComponent }); + if (headers) q.headers = headers; + if (body) { + if (typeof body === 'string') q.body = body; + else q.body = JSON.stringify(body); + } + + console.log('fetchCall', url); + + const response = await fetch(url, q); + + let rs; + try { + rs = await response.json(); + } catch (error) { + rs = await response.text(); + } + if (!response.ok) throw { error: rs }; + + return rs; +}; diff --git a/lib/seed/helpers/Request.ts b/lib/seed/helpers/Request.ts new file mode 100644 index 0000000..753c463 --- /dev/null +++ b/lib/seed/helpers/Request.ts @@ -0,0 +1,29 @@ +import { GetArgs } from '@seed/graphql/Request'; + +const DEFAULT_LIMIT = 100; +const DEFAULT_OFFSET = 0; + +export interface QueryPagination { + limit: number; + skip: number; + sort?: { + [k: string]: 1 | -1; + }; +} + +export const parsePaginationOptions = (obj?: GetArgs): QueryPagination => { + const limit = obj ? (obj.limit ? obj.limit : DEFAULT_LIMIT) : DEFAULT_LIMIT; + const skip = obj ? (obj.skip ? obj.skip : DEFAULT_OFFSET) : DEFAULT_OFFSET; + + if (obj && obj.sort) { + const sort = {}; + const sortArray = ('' + obj.sort).split(','); + for (let index = 0; index < sortArray.length; index++) { + const sortTmp = ('' + obj.sort).split(' '); + if (sortTmp[1].toLowerCase() === 'asc') (sort as any)[sortTmp[0]] = 1; + if (sortTmp[1].toLowerCase() === 'desc') (sort as any)[sortTmp[0]] = -1; + } + return { limit, skip, sort }; + } + return { limit, skip }; +}; diff --git a/lib/seed/helpers/StringHelper.ts b/lib/seed/helpers/StringHelper.ts new file mode 100644 index 0000000..3e6cbe0 --- /dev/null +++ b/lib/seed/helpers/StringHelper.ts @@ -0,0 +1,28 @@ +import { TranslatableComponent } from '@src/__components/components'; +import _ from 'lodash'; + +export const getAllTranslationsToString = (str: string, input: TranslatableComponent): string => { + for (const key in input) { + if (Object.prototype.hasOwnProperty.call(input, key)) { + str = str + '#' + input[key].toLowerCase(); + } + } + return str; +}; + +export const getSearchString = (...inputs: (string | TranslatableComponent)[]): string => { + let str = ''; + + for (let index = 0; index < inputs.length; index++) { + const element = inputs[index]; + if (_.isString(element)) str = str + '#' + element.toLowerCase(); + else str = getAllTranslationsToString(str, element); + } + + return str; +}; + +//escape regex => for special characters in email +export const escapeRegex = (regex) => { + return regex.replace(/[-[\]{}()*+?.,\\^$|#\s]/g, '\\$&'); +} \ No newline at end of file diff --git a/lib/seed/helpers/TestsHelper.ts b/lib/seed/helpers/TestsHelper.ts new file mode 100644 index 0000000..9b9fcf0 --- /dev/null +++ b/lib/seed/helpers/TestsHelper.ts @@ -0,0 +1,44 @@ +import fetch from 'node-fetch'; + +export const graphRequest = async (query, option): Promise => { + console.log('[TESTS] Query', query); + + let params; + if (option.authHeader) + params = { + body: JSON.stringify({ query }), + method: 'POST', + headers: { + authorization: option.authHeader, + 'Content-Type': 'application/json', + }, + }; + else + params = { + body: JSON.stringify({ query }), + method: 'POST', + headers: { + 'Content-Type': 'application/json', + }, + }; + + const response = await fetch(option.url, params); + if (!response.ok) { + return new Promise((resolve, reject) => { + response + .text() + .then((text) => { + try { + console.log('[TESTS] Error', JSON.parse(text)); + reject(JSON.parse(text)); + } catch (err) { + console.log('[TESTS] Error', text); + reject(text); + } + }) + .catch(reject); + }); + } + const json = await response.json(); + return json.data; +}; diff --git a/lib/seed/helpers/Utils.checks.ts b/lib/seed/helpers/Utils.checks.ts new file mode 100644 index 0000000..53ae52c --- /dev/null +++ b/lib/seed/helpers/Utils.checks.ts @@ -0,0 +1,14 @@ +import _ from 'lodash'; +import { newError } from './Error'; + +export const isAllThere = (listIds: string[], data: any[], modelName = 'modelName'): void => { + const ids = _.map(data, '_id'); + const nIds = _.difference(listIds, ids); + if (nIds.length > 0) throw newError(4041, { id: nIds.join(','), model: modelName }); + + return; +}; + +export const hasDuplicates = (array: any[]): boolean => { + return _.uniq(array).length !== array.length; +}; diff --git a/lib/seed/helpers/Utils.dates.intervals.ts b/lib/seed/helpers/Utils.dates.intervals.ts new file mode 100644 index 0000000..20a9726 --- /dev/null +++ b/lib/seed/helpers/Utils.dates.intervals.ts @@ -0,0 +1,88 @@ +import { DateTimeRangeComponent, AvailabilityComponent, DateRangeComponent, HourComponent } from '@seed/interfaces/components.dates'; +import _ from 'lodash'; +import { Interval, DateTime } from 'luxon'; + +export function checkIfDatesAreInAvailabilities( + dates: DateTimeRangeComponent[], + availabilities: AvailabilityComponent, + options?: { zone: string }, +): boolean { + const zone = options?.zone || process.env.TIMEZONE || 'utc'; + const intervalsExceptions = getExceptionsIntervalFromAvailabilities(availabilities, zone); + let isInException = false; + + if (intervalsExceptions.length > 0) { + isInException = _.some(dates, (d) => { + return checkIfDateRangeIsInIntervals(d, intervalsExceptions); + }); + } + + const intervalDates = getDatesIntervalFromAvailabilities(availabilities, zone); + let isInDates = true; + + if (intervalDates.length > 0) { + isInDates = _.every(dates, (d) => { + return checkIfDateRangeIsInIntervals(d, intervalDates); + }); + } + + return !isInException && isInDates; +} + +function checkIfDateRangeIsInIntervals(dateR: DateTimeRangeComponent, intervals: Interval[]): boolean { + const interval = Interval.fromDateTimes(dateR.startDate, dateR.endDate); + return _.some(intervals, (i) => i.overlaps(interval)); +} + +function getExceptionsIntervalFromAvailabilities(availabilities: AvailabilityComponent, zone: string): Interval[] { + const exceptions = availabilities.exceptions; + const intervalsExceptions: Interval[] = []; + + // Create all ezxceptions intervals + if (exceptions) { + exceptions.forEach((e) => { + createIntervalFromDatesAndHours(e.dates, zone, e.hours, intervalsExceptions); + }); + } + + return intervalsExceptions; +} + +function getDatesIntervalFromAvailabilities(availabilities: AvailabilityComponent, zone: string): Interval[] { + const dates = availabilities.dates; + const hours = availabilities.hours; + const datesExceptions: Interval[] = []; + + // Create all ezxceptions dates + if (dates) { + createIntervalFromDatesAndHours(dates, zone, hours, datesExceptions); + } + + return datesExceptions; +} + +function createIntervalFromDatesAndHours(dates: DateRangeComponent, zone: string, hours: HourComponent[] | undefined, datesExceptions: Interval[]) { + let dateStart = DateTime.fromJSDate(dates.startDate, { zone }); + const dateEnd = DateTime.fromJSDate(dates.endDate, { zone }); + + if (hours) { + while (dateStart < dateEnd) { + hours.forEach((h) => { + const hStart = parseInt(h.from.split(':')[0]); + const mStart = parseInt(h.from.split(':')[1]); + + const hEnd = parseInt(h.to.split(':')[0]); + const mEnd = parseInt(h.to.split(':')[1]); + + datesExceptions.push( + Interval.fromDateTimes(dateStart.set({ hour: hStart, minute: mStart }), dateStart.set({ hour: hEnd, minute: mEnd })), + ); + }); + dateStart = dateStart.plus({ day: 1 }); + } + + // if all day + } else { + datesExceptions.push(Interval.fromDateTimes(DateTime.fromJSDate(dates.startDate, { zone }), DateTime.fromJSDate(dates.endDate, { zone }))); + } +} diff --git a/lib/seed/helpers/Utils.dates.ts b/lib/seed/helpers/Utils.dates.ts new file mode 100644 index 0000000..c728862 --- /dev/null +++ b/lib/seed/helpers/Utils.dates.ts @@ -0,0 +1,310 @@ +import { + AvailabilityComponent, + DateRangeComponent, + DateTimeRangeComponent, + HourComponent, + SlotAvailablity, + SlotComponent, +} from '@seed/interfaces/components.dates'; +import _ from 'lodash'; +import { DateTime, Interval } from 'luxon'; + +interface hoursRange { + startHours: string; + startHoursInNumber: number; + endHours: string; + endHoursInNumber: number; +} +/* + breakTime: [ + ['11:00', '14:00'], + ['16:00', '18:00'], + ], +*/ + +export function getNumberOfWeekendDays(dateBegin: DateTime, dateEnd: DateTime): number { + let weekendDays = 0; + + let begin = dateBegin; + + while (begin <= dateEnd) { + if (begin.weekday == 6 || begin.weekday == 7) weekendDays = weekendDays + 1; + + begin = begin.plus({ day: 1 }); + } + + return weekendDays; +} + +function isInTimeInterval(slotTime: DateTime, slotInterval: number, timeRange: hoursRange[]): boolean { + const startTime = slotTime; + const endTime = slotTime.plus({ minutes: slotInterval }); + + const startTimeInNumber = startTime.hour * 60 + startTime.minute; + const endTimeInNumber = endTime.hour * 60 + endTime.minute; + + const isInInterval = timeRange.some((br) => { + return startTimeInNumber < br.endHoursInNumber && endTimeInNumber > br.startHoursInNumber; + // return (date.isSameOrAfter(startTime) && date.isSameOrBefore(endTime)) || (date.isSameOrAfter(startTime) && date.isSame(endTime)); + }); + + return isInInterval; +} + +export const getMinutesFromTimeInString = (str: string): number => { + const timeArray = str.split(':'); + return parseInt(timeArray[0]) * 60 + parseInt(timeArray[1]); +}; + +export const getDateNoTZ = (old: Date): Date => { + const userTimezoneOffset = old.getTimezoneOffset() * 60000; + return new Date(old.getTime() - userTimezoneOffset); +}; + +export const slotToDatesRange = (slot: SlotComponent): DateRangeComponent[] => { + const dates: DateRangeComponent[] = []; + const { startDate, endDate, startTime, endTime } = slot; + + const startTimes = startTime.split(':'); + const endTimes = endTime.split(':'); + + let beginDate = DateTime.fromJSDate(startDate).startOf('day'); + const stopDate = DateTime.fromJSDate(endDate).endOf('day'); + + while (beginDate < stopDate) { + dates.push({ + startDate: beginDate + .set({ hour: parseInt(startTimes[0]) }) + .set({ minute: parseInt(startTimes[1]) }) + .toJSDate(), + endDate: beginDate + .set({ hour: parseInt(endTimes[0]) }) + .set({ minute: parseInt(endTimes[1]) }) + // Remove one seconds for overlapping + .minus({ second: 1 }) + .toJSDate(), + }); + beginDate = beginDate.plus({ day: 1 }); + } + return dates; +}; + +export const getOneDaySlots = (input: { + day: DateTime; + dayUnavailability: DateTimeRangeComponent[]; + slotInfo: string[][]; + slotInterval: number; + slotDuration: number; + options?: { + zone?: string; + }; +}): string[] => { + const { day, dayUnavailability, slotInfo, slotDuration, slotInterval, options } = input; + + const zone = options?.zone || process.env.TIMEZONE || 'utc'; + + let currentDayMoment = day; + + // Check if currentDay = today + const currentDayStart = currentDayMoment.toISODate(); + const todayStart = DateTime.local().toISODate(); + + if (currentDayStart != todayStart) { + currentDayMoment = day.startOf('day'); + } + + const currentDayTimeInNumber = currentDayMoment.hour * 60 + currentDayMoment.minute; + + console.log('current day', currentDayMoment.toISO(), currentDayMoment.weekday); + + // Getting the slots for that day + const daySlot = slotInfo[day.weekday - 1]; + const currentDaySlots: { + startTime: DateTime; + endTime: DateTime; + }[] = []; + + if (daySlot && daySlot.length > 0) { + if (_.isArray(daySlot[0])) { + _.each(daySlot, (d) => { + currentDaySlots.push({ + startTime: DateTime.fromFormat(d[0], 'HH:mm', { zone }), + endTime: DateTime.fromFormat(d[1], 'HH:mm', { zone }), + }); + }); + } else { + currentDaySlots.push({ + startTime: DateTime.fromFormat(daySlot[0], 'HH:mm', { zone }), + endTime: DateTime.fromFormat(daySlot[1], 'HH:mm', { zone }), + }); + } + } + + const thisDaysUnavailabilities: { + startHours: string; + startHoursInNumber: number; + endHours: string; + endHoursInNumber: number; + }[] = getUnavailabilityOfThisDay(dayUnavailability, currentDayMoment); + + const currentDateSlots: string[] = []; + console.log('thisDaysUnavailabilities', thisDaysUnavailabilities); + + currentDaySlots.forEach((element) => { + let { startTime, endTime } = element; + + while (startTime < endTime) { + if ( + !(startTime.hour * 60 + startTime.minute < currentDayTimeInNumber) && + !isInTimeInterval(startTime, slotDuration, thisDaysUnavailabilities) && + startTime.hour * 60 + startTime.minute + slotDuration <= endTime.hour * 60 + endTime.minute + ) { + currentDateSlots.push(startTime.toFormat('HH:mm')); + } + startTime = startTime.plus({ minutes: slotInterval }); + } + }); + + return currentDateSlots; +}; + +export const getMultipleDaysSlots = (input: { + datesWanted: DateRangeComponent; + datesUnavailable: DateTimeRangeComponent[]; + slotInterval: number; + slotInfo: string[][]; + slotDuration: number; + options?: { + zone?: string; + }; +}): SlotAvailablity[] => { + const returnSlots: SlotAvailablity[] = []; + const { datesWanted, datesUnavailable, slotInterval, slotInfo, slotDuration, options } = input; + const { startDate, endDate } = datesWanted; + + const zone = options?.zone || process.env.TIMEZONE || 'utc'; + + let beginDate = DateTime.fromJSDate(startDate, { zone }); + const stopDate = DateTime.fromJSDate(endDate, { zone }); + + // Two loops + // Looping into the wanted dates and keeping an index for the slotInfo + while (beginDate < stopDate) { + // Get the slot info from the array of array + const slots = getOneDaySlots({ + day: beginDate, + dayUnavailability: datesUnavailable, + slotInfo, + slotInterval, + slotDuration, + options: { zone }, + }); + + // Adding it to the array + returnSlots.push({ + date: beginDate.toJSDate(), + slots, + }); + + beginDate = beginDate.plus({ day: 1 }).startOf('day'); + } + + return returnSlots; +}; + +export function getUnavailabilityOfThisDay(dayUnavailability: DateTimeRangeComponent[], currentDayMoment: DateTime) { + const thisDaysUnavailabilities: { + startHours: string; + startHoursInNumber: number; + endHours: string; + endHoursInNumber: number; + }[] = []; + + dayUnavailability.forEach((element) => { + const startDate = element.startDate; + const endDate = element.endDate; + + // If there are some reservations in that day or in multiple days + const isSameThanStartDate = currentDayMoment.hasSame(startDate, 'day'); + const isSameThanEndDate = currentDayMoment.hasSame(endDate, 'day'); + + const startHours = startDate.toFormat('HH:mm'); + const endHours = endDate.toFormat('HH:mm'); + const starts = startHours.split(':'); + const ends = endHours.split(':'); + + const startDateDay = startDate.startOf('day'); + const endDateDay = startDate.endOf('day'); + + const startHoursInNumber = parseInt(starts[0]) * 60 + parseInt(starts[1]); + const endHoursInNumber = parseInt(ends[0]) * 60 + parseInt(ends[1]); + + // If the reservation only last a day + if (isSameThanStartDate && isSameThanEndDate) { + thisDaysUnavailabilities.push({ + startHours, + endHours, + startHoursInNumber, + endHoursInNumber, + }); + } + + // If the reservation last multiple days + else if (currentDayMoment >= startDateDay && currentDayMoment <= endDateDay) { + // check if start date + if (isSameThanStartDate) + thisDaysUnavailabilities.push({ + startHours, + endHours: '23:59', + startHoursInNumber, + endHoursInNumber: 23 * 60 + 59, + }); + else if (isSameThanEndDate) + thisDaysUnavailabilities.push({ + startHours: '00:00', + endHours, + startHoursInNumber: 0, + endHoursInNumber, + }); + else + thisDaysUnavailabilities.push({ + startHours: '00:00', + endHours: '23:59', + startHoursInNumber: 0, + endHoursInNumber: 23 * 60 + 59, + }); + } + }); + return thisDaysUnavailabilities; +} + +export function isTimeSlotInDates(dates: DateTimeRangeComponent[], startTime: string, endTime: string) { + const startTimesInMinutes = getMinutesFromTimeInString(startTime); + const endTimesInMinutes = getMinutesFromTimeInString(endTime); + + // Check for each date if the minutes are in it + return dates.some((br) => { + const brStartTimeInMinutes = br.startDate.hour * 60 + br.startDate.minute; + const brEndTimeInMinutes = br.endDate.hour * 60 + br.endDate.minute; + return startTimesInMinutes < brEndTimeInMinutes && endTimesInMinutes > brStartTimeInMinutes; + }); +} + +export function converDateRangeToDateTime( + dates: DateRangeComponent[], + options?: { + zone?: string; + }, +): DateTimeRangeComponent[] { + const results: DateTimeRangeComponent[] = []; + const zone = options?.zone || process.env.TIMEZONE || 'utc'; + + dates.forEach((element) => { + results.push({ + startDate: DateTime.fromJSDate(element.startDate, { zone }), + endDate: DateTime.fromJSDate(element.endDate, { zone }), + }); + }); + + return results; +} diff --git a/lib/seed/helpers/Utils.string.ts b/lib/seed/helpers/Utils.string.ts new file mode 100644 index 0000000..49f5574 --- /dev/null +++ b/lib/seed/helpers/Utils.string.ts @@ -0,0 +1,38 @@ +import _ from 'lodash'; + +export class UtilStringHelper { + interpolate = function( + string: string, + data: any, + replace?: { + withEol?: boolean; + }, + ): any { + let finalString = string; + + if (!replace || !replace.withEol) finalString = finalString = finalString.replace(/\s{2,}/g, ''); + // finalString.replace(/(?:\r\n|\r|\n)/g, '
'); + // // else + // finalString = finalString.replace(/\s{2,}/g, ''); // finalString.replace(/(?:\r\n|\r|\n)/g, ''); + + try { + const compiled = _.template(finalString); + if (finalString.includes('${data.')) return compiled({ data }); + return compiled(data); + } catch (error) { + console.error(error); + + const names = Object.keys(data); + const vals = Object.values(data) as any; + for (let index = 0; index < names.length; index++) { + finalString = finalString.replace('${' + names[index] + '}', vals[index]); + } + return finalString; + // return new Function(...names, "return String.raw`${this}`")(...vals); + } + }; + + removeTabAndReturn = function(string: string): string { + return string.replace(/[\n\t]+/g, ''); + }; +} diff --git a/lib/seed/helpers/Utils.ts b/lib/seed/helpers/Utils.ts new file mode 100644 index 0000000..4478078 --- /dev/null +++ b/lib/seed/helpers/Utils.ts @@ -0,0 +1,64 @@ +import _ from 'lodash'; +import { UtilStringHelper } from './Utils.string'; + +const StringHelper = new UtilStringHelper(); + +export const interpolate = StringHelper.interpolate; + +export const sleep = async (ms: number): Promise => { + return new Promise((resolve) => { + setTimeout(resolve, ms); + }); +}; + +export const promiseAll = async (promises: Promise[]): Promise => { + const results = await Promise.allSettled(promises); + + results.forEach((element) => { + if (element.status == 'rejected') console.error('[ASYNC - ERROR]', element.reason); + }); + + console.log('[PROMISE ALL - DONE]'); + + return results; +}; + +// eslint-disable-next-line @typescript-eslint/explicit-function-return-type +const isNullBlankOrUndefined = function(o) { + return typeof o === 'undefined' || o == null || o === ''; +}; +/** + * Deep diff between two object, using lodash + * @param {Object} object Object compared + * @param {Object} base Object to compare with + * @return {Object} Return a new object who represent the diff + */ +export const differenceBetweenObject = (object, base, ignoreBlanks = false): any => { + if (!_.isObject(object) || _.isDate(object) || Array.isArray(object)) return object; // special case dates & array + return _.transform(object as any, (result, value, key) => { + if (!_.isEqual(value, base[key])) { + if (ignoreBlanks && isNullBlankOrUndefined(value) && isNullBlankOrUndefined(base[key])) return; + result[key] = _.isObject(value) && _.isObject(base[key]) ? differenceBetweenObject(value, base[key]) : value; + } + }); +}; + +export const generateCodeNumber = () => { + return Math.floor(100000 + Math.random() * 900000); +}; + +export const clog = (data: any): void => { + console.log(JSON.stringify(data, null, 2)); +}; + +export const nullToUndefined = (value: any): any | undefined => { + if (_.isString(value) || _.isBoolean(value) || _.isDate(value) || _.isFinite(value) || _.isInteger(value)) return value; + + if (_.isArray(value)) return (value as any[]).map(nullToUndefined); + + if (_.isObject(value)) return _.mapValues(value as any, nullToUndefined); + + if (value === null) return undefined; + + return value; +}; diff --git a/lib/seed/interfaces/components.dates.ts b/lib/seed/interfaces/components.dates.ts new file mode 100644 index 0000000..5a9b421 --- /dev/null +++ b/lib/seed/interfaces/components.dates.ts @@ -0,0 +1,114 @@ +import { InputType, Int, Field, ArgsType, ObjectType } from 'type-graphql'; +import { IsDate, IsMilitaryTime } from 'class-validator'; +import { DateTime } from 'luxon'; + +@ObjectType() +@InputType('ScheduleInput') +export class ScheduleComponent { + @Field(() => [[String]], { nullable: true }) + mon?: string[][]; + @Field(() => [[String]], { nullable: true }) + tue?: string[][]; + @Field(() => [[String]], { nullable: true }) + wed?: string[][]; + @Field(() => [[String]], { nullable: true }) + thu?: string[][]; + @Field(() => [[String]], { nullable: true }) + fri?: string[][]; + @Field(() => [[String]], { nullable: true }) + sat?: string[][]; + @Field(() => [[String]], { nullable: true }) + sun?: string[][]; +} + +@ObjectType() +@InputType('HoursInput') +export class HourComponent { + @Field() + from: string; + @Field() + to: string; +} + +@ObjectType() +@InputType('DateRangeComponentInput') +@ArgsType() +export class DateRangeComponent { + @Field() + @IsDate() + startDate: Date; + @Field() + @IsDate() + endDate: Date; +} + +export class DateTimeRangeComponent { + startDate: DateTime; + endDate: DateTime; +} + +@ObjectType() +@InputType('SlotAvailablityInput') +export class SlotAvailablity { + @Field() + @IsDate() + date: Date; + + @Field(() => [String]) + slots: string[]; +} + +@ArgsType() +export class SlotAvailablityRequest extends DateRangeComponent { + @Field(() => Int) + interval: number; + + @Field(() => Int) + duration: number; +} + +@ObjectType() +@InputType('SlotComponentInput') +export class SlotComponent { + @Field() + @IsDate() + startDate: Date; + + @Field() + @IsDate() + endDate: Date; + + @Field() + @IsMilitaryTime() + startTime: string; + + @Field() + @IsMilitaryTime() + endTime: string; +} + +@ObjectType() +@InputType('BaseAvailabilityComponentInput') +export class BaseAvailabilityComponent { + @Field(() => DateRangeComponent) + dates: DateRangeComponent; + @Field(() => [HourComponent], { nullable: true }) + hours?: HourComponent[]; +} + +@ObjectType() +@InputType('DateExceptionComponentInput') +export class DateExceptionComponent extends BaseAvailabilityComponent { + @Field({ nullable: true }) + allDay?: boolean; +} + +@ObjectType() +@InputType('AvailabilityInput') +export class AvailabilityComponent extends BaseAvailabilityComponent { + @Field(() => [DateExceptionComponent], { nullable: true }) + exceptions?: DateExceptionComponent[]; + + @Field({ nullable: true }) + noWeekend?: boolean; +} diff --git a/lib/seed/interfaces/components.errors.ts b/lib/seed/interfaces/components.errors.ts new file mode 100644 index 0000000..7dc7fa0 --- /dev/null +++ b/lib/seed/interfaces/components.errors.ts @@ -0,0 +1,99 @@ +/* + ███████╗███╗ ██╗██╗ ██╗███╗ ███╗ + ██╔════╝████╗ ██║██║ ██║████╗ ████║ + █████╗ ██╔██╗ ██║██║ ██║██╔████╔██║ + ██╔══╝ ██║╚██╗██║██║ ██║██║╚██╔╝██║ + ███████╗██║ ╚████║╚██████╔╝██║ ╚═╝ ██║ + ╚══════╝╚═╝ ╚═══╝ ╚═════╝ ╚═╝ ╚═╝ +*/ + +/* + ██████╗ ██████╗ ███╗ ███╗██████╗ ██████╗ ███╗ ██╗███████╗███╗ ██╗████████╗███████╗ + ██╔════╝██╔═══██╗████╗ ████║██╔══██╗██╔═══██╗████╗ ██║██╔════╝████╗ ██║╚══██╔══╝██╔════╝ + ██║ ██║ ██║██╔████╔██║██████╔╝██║ ██║██╔██╗ ██║█████╗ ██╔██╗ ██║ ██║ ███████╗ + ██║ ██║ ██║██║╚██╔╝██║██╔═══╝ ██║ ██║██║╚██╗██║██╔══╝ ██║╚██╗██║ ██║ ╚════██║ + ╚██████╗╚██████╔╝██║ ╚═╝ ██║██║ ╚██████╔╝██║ ╚████║███████╗██║ ╚████║ ██║ ███████║ + ╚═════╝ ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═════╝ ╚═╝ ╚═══╝╚══════╝╚═╝ ╚═══╝ ╚═╝ ╚══════╝ +*/ + +export const seedErrorConfig = { + /* HTTP like errors */ + 404: 'The document was not found', + 4040: 'The ressource ${obj.ressourceId} for the model ${obj.ressourceModel} does not exists', + 4041: 'The id "${id}" for the model "${model}" was not found', + + /* ************* */ + /* General & DB */ + /* ************* */ + 1000: 'General database error', + 1001: 'The document was not deleted', + 1002: 'This action is not allowed', + + /* ************* */ + /* Permissions - Accounts & Organisation */ + /* ************* */ + + // Accounts + 2000: 'Your permissions level are not allowed', + 2002: 'Authorization headers are required for this action', + 2003: 'The account is already registered', + 2004: 'We have no records of your account in our database', + 2005: 'There was an error creating your account', + 2006: 'The password confirmation you entered is incorrect', + 20060: 'The old confirmation you entered is incorrect', + 2007: 'There was an error updating your permission', + + // Firebase + 2010: 'Your login failed. Please verify your credentials', + 2011: 'We had issues with your token. Please retry', + 2012: 'We had issues creating your account. Please retry', + 2013: 'There was an issue with the autentication provider', + 2015: "Linking your account didn't work. Please contact administrator", + + // Organisations + 2100: 'Your permissions on the organisation level are not allowed', + 2101: "Your account doesn't operate in that organisation", + 2102: 'Organisation headers are required for this action', + + // Api keys + 2200: 'Your api keys are not in our system', + 2202: 'Api keys are required for this action', + + // Codes + 2300: 'The code you entered does not exists', + 2301: 'The code you entered has expired', + 2302: 'Too many attempts, please retry in 15 mins', + + /* ************* */ + /* Security */ + /* ************* */ + 2400: 'You need to complete security protocols before doing that action', + + /* ************* */ + /* DataValidation */ + /* ************* */ + + 3000: 'There are some inputs missing', + 3001: 'This is a validation error', + + 3005: 'There was an error with the errors range time', + + // -- Forms + 3100: 'Items are required for that form type', + 3101: 'The form has duplicate name entries', + 3102: "The section '${sName}' for this form doesn't exists", + 3103: "The question '${qName}' for the section '${sName}' for this form doesn't exists", + 3104: "The answers of the section '${sectionName}' doesn't exists", + 31040: "The answer ${fieldName} of the section '${sectionName}' doesn't exists", + 3105: 'The answer "${name}" for this field "${formName}" is not acceptable', + + 3106: "The tab ${tabName} for this form doesn't exists", + 3107: 'Please select a section to answer', + 3108: 'Some input are missing, ${name} for section ${sectionName}', + 3109: 'Data integrity breach, ${name} should be ${type}', + 3110: 'Only one answer (key) please', + + // -- Files + 3200: 'System cannot read the extension of the file', + 3201: 'The file extensions are not allowed - allowed : ${allowedMime}', +}; diff --git a/lib/seed/interfaces/components.geo.ts b/lib/seed/interfaces/components.geo.ts new file mode 100644 index 0000000..c466b31 --- /dev/null +++ b/lib/seed/interfaces/components.geo.ts @@ -0,0 +1,192 @@ +import { InputType, Int, Field, ArgsType, ObjectType, Args } from 'type-graphql'; + +import { Max, IsNotEmpty } from 'class-validator'; + +import { registerEnumType } from 'type-graphql'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { TranslatableComponent } from '@src/__components/components'; +import { ModelCollectionEnum } from '@src/__indexes/__collections'; +import { IsRefExist } from '@seed/engine/decorators/db.guard'; +import { ScheduleComponent } from './components.dates'; + +/* + ███████╗███╗ ██╗██╗ ██╗███╗ ███╗ + ██╔════╝████╗ ██║██║ ██║████╗ ████║ + █████╗ ██╔██╗ ██║██║ ██║██╔████╔██║ + ██╔══╝ ██║╚██╗██║██║ ██║██║╚██╔╝██║ + ███████╗██║ ╚████║╚██████╔╝██║ ╚═╝ ██║ + ╚══════╝╚═╝ ╚═══╝ ╚═════╝ ╚═╝ ╚═╝ +*/ + +/* + ██████╗ ██████╗ ███╗ ███╗██████╗ ██████╗ ███╗ ██╗███████╗███╗ ██╗████████╗███████╗ + ██╔════╝██╔═══██╗████╗ ████║██╔══██╗██╔═══██╗████╗ ██║██╔════╝████╗ ██║╚══██╔══╝██╔════╝ + ██║ ██║ ██║██╔████╔██║██████╔╝██║ ██║██╔██╗ ██║█████╗ ██╔██╗ ██║ ██║ ███████╗ + ██║ ██║ ██║██║╚██╔╝██║██╔═══╝ ██║ ██║██║╚██╗██║██╔══╝ ██║╚██╗██║ ██║ ╚════██║ + ╚██████╗╚██████╔╝██║ ╚═╝ ██║██║ ╚██████╔╝██║ ╚████║███████╗██║ ╚████║ ██║ ███████║ + ╚═════╝ ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═════╝ ╚═╝ ╚═══╝╚══════╝╚═╝ ╚═══╝ ╚═╝ ╚══════╝ +*/ + +/* + * ADDRESS + */ +@InputType('LocComponentInput') +@ObjectType() +export class LocComponent { + @Field() + type: string; + @Field(() => [Number]) + coordinates: number[]; +} + +@InputType('AddressInput') +@ObjectType() +export class AddressComponent { + @Field({ nullable: true }) + number?: string; + @Field({ nullable: true }) + street?: string; + @Field({ nullable: true }) + streetBis?: string; + @Field({ nullable: true }) + floor?: string; + @Field({ nullable: true }) + box?: string; + @Field({ nullable: true }) + zip?: string; + @Field({ nullable: true }) + state?: string; + @Field() + city: string; + @Field() + country: string; + + returnFullAddress() { + return `${this.number} ${this.street}, ${this.zip} ${this.city} ${this.country}` + } +} + +@InputType('AddressStrictInput') +@ObjectType() +export class AddressStrictComponent { + @Field() + number: string; + @Field() + street: string; + @Field() + zip: string; + @Field() + city: string; + @Field() + country: string; + + @Field({ nullable: true }) + streetBis?: string; + @Field({ nullable: true }) + floor?: string; + @Field({ nullable: true }) + box?: string; + @Field({ nullable: true }) + state?: string; + + + returnFullAddress() { + return `${this.number} ${this.street}, ${this.zip} ${this.city} ${this.country}` + } +} + +@InputType('PlaceInput') +@ObjectType() +export class PlaceComponent { + @Field() + placeId: string; + @Field({ nullable: true }) + address: AddressComponent; + @Field(() => LocComponent) + loc: LocComponent; + @Field() + formattedAddress: string; +} + +@InputType('PlaceComponentOptionnalInput') +@ObjectType() +export class PlaceComponentOptionnal { + @Field({ nullable: true }) + placeId?: string; + @Field(() => AddressComponent, { nullable: true }) + address: AddressComponent; + @Field(() => LocComponent, { nullable: true }) + loc?: LocComponent; + @Field({ nullable: true }) + formattedAddress?: string; +} + +@InputType('PlaceComponentOptionnalWithAddressInput') +@ObjectType() +export class PlaceComponentOptionnalWithAddress { + @Field({ nullable: true }) + placeId?: string; + @Field(() => AddressStrictComponent) + address: AddressStrictComponent; + @Field(() => LocComponent, { nullable: true }) + loc?: LocComponent; + @Field({ nullable: true }) + formattedAddress?: string; +} + +@InputType('GeolocSearchInput') +@ObjectType() +@ArgsType() +export class GeolocSearchComponent { + @Field() + longitude: number; + @Field() + latitude: number; + + @Max(10000) + @Field(() => Int, { nullable: true }) + radius = 5000; +} + +@InputType('GeolocAddressSearchInput') +@ObjectType() +export class GeolocAddressSearchComponent { + @Field() + formattedAddress: string; + + @Max(10000) + @Field(() => Int, { nullable: true }) + radius = 5000; +} + +@InputType('GeolocPlaceSearchInput') +@ObjectType() +export class GeolocPlaceSearchComponent { + @Field() + placeId: string; + + @Max(10000) + @Field(() => Int, { nullable: true }) + radius = 5000; +} + +@ObjectType() +export class GeolocDistComponent { + @Field({ nullable: true }) + calculated?: number; + @Field(() => LocComponent, { nullable: true }) + location?: LocComponent; +} + +@InputType('PlaceContactInformationInput') +@ObjectType() +export class PlaceContactInformation { + @Field() + phoneNumber: string; + + @Field() + email: string; + + @Field(() => ScheduleComponent) + openingHours: ScheduleComponent; +} diff --git a/lib/seed/interfaces/components.notifications.ts b/lib/seed/interfaces/components.notifications.ts new file mode 100644 index 0000000..5fa0091 --- /dev/null +++ b/lib/seed/interfaces/components.notifications.ts @@ -0,0 +1,24 @@ +/* + ███████╗███╗ ██╗██╗ ██╗███╗ ███╗ + ██╔════╝████╗ ██║██║ ██║████╗ ████║ + █████╗ ██╔██╗ ██║██║ ██║██╔████╔██║ + ██╔══╝ ██║╚██╗██║██║ ██║██║╚██╔╝██║ + ███████╗██║ ╚████║╚██████╔╝██║ ╚═╝ ██║ + ╚══════╝╚═╝ ╚═══╝ ╚═════╝ ╚═╝ ╚═╝ +*/ + +export enum EngineNotificationEnum { + 'emailCodeSignIn' = 'emailCodeSignIn', + 'magicLink' = 'magicLink', + 'resetPassword' = 'resetPassword', + 'adminAccountAddOne' = 'adminAccountAddOne', +} + +/* + ██████╗ ██████╗ ███╗ ███╗██████╗ ██████╗ ███╗ ██╗███████╗███╗ ██╗████████╗███████╗ + ██╔════╝██╔═══██╗████╗ ████║██╔══██╗██╔═══██╗████╗ ██║██╔════╝████╗ ██║╚══██╔══╝██╔════╝ + ██║ ██║ ██║██╔████╔██║██████╔╝██║ ██║██╔██╗ ██║█████╗ ██╔██╗ ██║ ██║ ███████╗ + ██║ ██║ ██║██║╚██╔╝██║██╔═══╝ ██║ ██║██║╚██╗██║██╔══╝ ██║╚██╗██║ ██║ ╚════██║ + ╚██████╗╚██████╔╝██║ ╚═╝ ██║██║ ╚██████╔╝██║ ╚████║███████╗██║ ╚████║ ██║ ███████║ + ╚═════╝ ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═════╝ ╚═╝ ╚═══╝╚══════╝╚═╝ ╚═══╝ ╚═╝ ╚══════╝ +*/ diff --git a/lib/seed/interfaces/components.texts.ts b/lib/seed/interfaces/components.texts.ts new file mode 100644 index 0000000..7dd5c1e --- /dev/null +++ b/lib/seed/interfaces/components.texts.ts @@ -0,0 +1,46 @@ +import { InputType, Int, Field, ArgsType, ObjectType, Args } from 'type-graphql'; + +import { Max, IsDate, IsMilitaryTime, IsNotEmpty } from 'class-validator'; + +import { registerEnumType } from 'type-graphql'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { TranslatableComponent } from '@src/__components/components'; +import { ModelCollectionEnum } from '@src/__indexes/__collections'; +import { IsRefExist } from '@seed/engine/decorators/db.guard'; +import { DateTime } from 'luxon'; + +/* + ███████╗███╗ ██╗██╗ ██╗███╗ ███╗ + ██╔════╝████╗ ██║██║ ██║████╗ ████║ + █████╗ ██╔██╗ ██║██║ ██║██╔████╔██║ + ██╔══╝ ██║╚██╗██║██║ ██║██║╚██╔╝██║ + ███████╗██║ ╚████║╚██████╔╝██║ ╚═╝ ██║ + ╚══════╝╚═╝ ╚═══╝ ╚═════╝ ╚═╝ ╚═╝ +*/ + +/* + ██████╗ ██████╗ ███╗ ███╗██████╗ ██████╗ ███╗ ██╗███████╗███╗ ██╗████████╗███████╗ + ██╔════╝██╔═══██╗████╗ ████║██╔══██╗██╔═══██╗████╗ ██║██╔════╝████╗ ██║╚══██╔══╝██╔════╝ + ██║ ██║ ██║██╔████╔██║██████╔╝██║ ██║██╔██╗ ██║█████╗ ██╔██╗ ██║ ██║ ███████╗ + ██║ ██║ ██║██║╚██╔╝██║██╔═══╝ ██║ ██║██║╚██╗██║██╔══╝ ██║╚██╗██║ ██║ ╚════██║ + ╚██████╗╚██████╔╝██║ ╚═╝ ██║██║ ╚██████╔╝██║ ╚████║███████╗██║ ╚████║ ██║ ███████║ + ╚═════╝ ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═════╝ ╚═╝ ╚═══╝╚══════╝╚═╝ ╚═══╝ ╚═╝ ╚══════╝ +*/ + +@ObjectType() +@InputType('GenericValueLabelComponentInput') +export class GenericValueLabelComponent { + @Field() + value: string; + @Field() + label: TranslatableComponent; +} + +@ObjectType() +@InputType('GenericTitleSubtitleComponentInput') +export class GenericTitleSubtitleComponent { + @Field(() => TranslatableComponent, { nullable: true }) + title: TranslatableComponent; + @Field(() => TranslatableComponent, { nullable: true }) + subtitle: TranslatableComponent; +} diff --git a/lib/seed/interfaces/components.ts b/lib/seed/interfaces/components.ts new file mode 100644 index 0000000..ee737bf --- /dev/null +++ b/lib/seed/interfaces/components.ts @@ -0,0 +1,318 @@ +import { InputType, Int, Field, ArgsType, ObjectType, Args } from 'type-graphql'; + +import { Max, IsNotEmpty } from 'class-validator'; + +import { registerEnumType } from 'type-graphql'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { TranslatableComponent } from '@src/__components/components'; +import { ModelCollectionEnum } from '@src/__indexes/__collections'; +import { IsRefExist } from '@seed/engine/decorators/db.guard'; + +/* + ███████╗███╗ ██╗██╗ ██╗███╗ ███╗ + ██╔════╝████╗ ██║██║ ██║████╗ ████║ + █████╗ ██╔██╗ ██║██║ ██║██╔████╔██║ + ██╔══╝ ██║╚██╗██║██║ ██║██║╚██╔╝██║ + ███████╗██║ ╚████║╚██████╔╝██║ ╚═╝ ██║ + ╚══════╝╚═╝ ╚═══╝ ╚═════╝ ╚═╝ ╚═╝ +*/ + +export enum SettingsType { + emails = 'emails', + ac = 'ac', + env = 'env', + instagram = 'instagram', + errors = 'errors', + others = 'others', +} +registerEnumType(SettingsType, { + name: 'SettingsType', +}); + +export enum AccountGenderEnum { + m = 'm', + f = 'f', + other = 'other', +} +registerEnumType(AccountGenderEnum, { + name: 'AccountGenderEnum', +}); + +export enum deviceOs { + ios = 'ios', + android = 'android', +} +registerEnumType(deviceOs, { + name: 'deviceOs', +}); + +/* + * Permissions + */ + +export enum PermissionType { + r = 'r', + w = 'w', + d = 'd', +} +registerEnumType(PermissionType, { + name: 'PermissionType', +}); + +/* + * Date stuff + */ + +export enum RangeType { + strict = 'strict', // IS << A - B >> IE + intersect = 'intersect', // IS << A - IE << B || + intersectLarge = 'intersectLarge', + included = 'included', // IS << A - B >> IE +} +registerEnumType(RangeType, { + name: 'RangeType', +}); + +export enum DateTabType { + today = 'today', + tomorrow = 'tomorrow', + upcomming = 'upcomming', + past = 'past', +} +registerEnumType(DateTabType, { + name: 'DateTabType', +}); + +export enum SizeTypeEnum { + cm = 'cm', + inch = 'inch', +} +registerEnumType(SizeTypeEnum, { + name: 'SizeTypeEnum', +}); + +export enum WeightTypeEnum { + lbs = 'lbs', + g = 'g', +} +registerEnumType(WeightTypeEnum, { + name: 'WeightTypeEnum', +}); + +export enum QualityEnum { + verygood = 'verygood', + good = 'good', + neutral = 'neutral', + poor = 'poor', + verypoor = 'verypoor', +} +registerEnumType(QualityEnum, { + name: 'QualityEnum', +}); + +export enum StatusEnum { + draft = 'draft', + processing = 'processing', + validated = 'validated', + cancelled = 'cancelled', + expired = 'expired', + +} +registerEnumType(StatusEnum, { + name: 'StatusEnum', +}); + +// export enum IntervalEnum { +// '15' = '15', +// '30' = '30', +// '45' = '45', +// '60' = '60', +// '90' = '90', +// '120' = '120', +// } +// registerEnumType(IntervalEnum, { +// name: 'IntervalEnum', +// }); + +/* + ██████╗ ██████╗ ███╗ ███╗██████╗ ██████╗ ███╗ ██╗███████╗███╗ ██╗████████╗███████╗ + ██╔════╝██╔═══██╗████╗ ████║██╔══██╗██╔═══██╗████╗ ██║██╔════╝████╗ ██║╚══██╔══╝██╔════╝ + ██║ ██║ ██║██╔████╔██║██████╔╝██║ ██║██╔██╗ ██║█████╗ ██╔██╗ ██║ ██║ ███████╗ + ██║ ██║ ██║██║╚██╔╝██║██╔═══╝ ██║ ██║██║╚██╗██║██╔══╝ ██║╚██╗██║ ██║ ╚════██║ + ╚██████╗╚██████╔╝██║ ╚═╝ ██║██║ ╚██████╔╝██║ ╚████║███████╗██║ ╚████║ ██║ ███████║ + ╚═════╝ ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═════╝ ╚═╝ ╚═══╝╚══════╝╚═╝ ╚═══╝ ╚═╝ ╚══════╝ +*/ + +@ObjectType() +export class EnginePathComponent { + @Field(() => ModelCollectionEnum) + @IsNotEmpty() + ressourceModel: ModelCollectionEnum | string; + + @Field() + @IsNotEmpty() + // @IsRefExist() + ressourceId: string; +} + +/* + * AUTH + * + */ + +@ObjectType() +export class FirebaseTokenResult { + @Field() + localId: string; + @Field({ nullable: true }) + email?: string; + @Field({ nullable: true }) + displayName?: string; + @Field() + idToken: string; + @Field({ nullable: true }) + registered?: boolean; + @Field() + refreshToken: string; + @Field() + expiresIn: string; +} + +/* + * MISC + */ +@InputType('PositionInInput') +export class PositionInInterface { + @Field() + _id: string; + + @Field(() => Int) + order: number; +} +[]; + +// @ObjectType() +// @InputType('TranslatableInput') +// export class TranslatableComponent { +// @Field() +// en: string; +// } + +/* + * RESULTS + */ + +@ObjectType('IdResult') +export class IdResult { + @Field() + _id: string; +} + +@ObjectType() +export class SimpleResult { + @Field() + message: string; +} + +/* + * DEVICES + */ + +@ObjectType() +@InputType('DeviceInput') +export class DeviceComponent { + @Field() + uid: string; + @Field(() => deviceOs) + os: deviceOs; + + @Field() + pushToken: string; + @Field() + deviceName: string; +} + +/* + * IMAGES + */ + +@ObjectType() +@InputType('ImageInput') +export default class ImageComponent { + @Field({ nullable: true }) + title?: string; + @Field({ nullable: true }) + fileType?: string; + + @Field() + large: string; + + @Field({ nullable: true }) + medium?: string; + + @Field({ nullable: true }) + small?: string; +} + +@ObjectType() +@InputType('NewPermissionInputC') +export class NewPermissionInput { + @Field(() => PermissionType) + permissionType: PermissionType; + + @Field(() => AccountTypeEnum) + permission: AccountTypeEnum; +} + +@ObjectType() +@InputType('SEOInput') +export class SEOField { + @Field(() => TranslatableComponent) + title: TranslatableComponent; + @Field(() => TranslatableComponent) + description: TranslatableComponent; + @Field(() => TranslatableComponent, { nullable: true }) + keywords?: TranslatableComponent; + @Field(() => ImageComponent, { nullable: true }) + thumbnail?: ImageComponent; +} + +@ObjectType() +@InputType('SEOSimpleInput') +export class SEOSimpleField { + @Field() + title: string; + @Field() + description: string; + @Field({ nullable: true }) + keywords?: string; + @Field(() => ImageComponent, { nullable: true }) + thumbnail?: ImageComponent; +} + +@ObjectType() +@InputType('SizeComponentInput') +export class SizeComponent { + @Field() + unit: SizeTypeEnum; + @Field() + value: number; +} + +@ObjectType() +@InputType('WeightComponentInput') +export class WeightComponent { + @Field() + unit: WeightTypeEnum; + @Field() + value: number; +} +@ObjectType() +@InputType('CustomTagsInput') +export class CustomTagsInterface { + @Field() + key: string; + + @Field() + value: string; +} diff --git a/lib/seed/interfaces/context.ts b/lib/seed/interfaces/context.ts new file mode 100644 index 0000000..4275002 --- /dev/null +++ b/lib/seed/interfaces/context.ts @@ -0,0 +1,22 @@ +import { APIGatewayProxyEvent, Context } from 'aws-lambda'; +import AccountModel from '@src/accounts/account.model'; + +export interface ApolloContext { + event: APIGatewayProxyEvent; + context: Context; + ctx: { + organisationId: string | null; + user: AccountModel; + loaders: any; + noOrganisationCheck?: boolean; + noPermissionCheck?: boolean; + }; +} + +export interface ApolloContextLoadersOnly { + event: APIGatewayProxyEvent; + context: Context; + ctx: { + loaders: any; + }; +} diff --git a/lib/seed/interfaces/permission.ts b/lib/seed/interfaces/permission.ts new file mode 100644 index 0000000..eec122d --- /dev/null +++ b/lib/seed/interfaces/permission.ts @@ -0,0 +1,24 @@ +import { AccountTypeEnum } from '@src/accounts/account.components'; + +export interface Permission { + c: (AccountTypeEnum | string)[]; + r: (AccountTypeEnum | string)[]; + w: (AccountTypeEnum | string)[]; + d: (AccountTypeEnum | string)[]; +} +export interface PermissionDB { + r: AccountTypeEnum[]; + w: AccountTypeEnum[]; + d: AccountTypeEnum[]; +} + +export interface PermissionIN { + operation: 'add' | 'remove'; + field: 'r' | 'w' | 'd'; + newPerm: string; +} + +export interface PermissionInfo { + accountId: string; + accountGroup: string[]; +} diff --git a/lib/seed/interfaces/response.ts b/lib/seed/interfaces/response.ts new file mode 100644 index 0000000..719a7f1 --- /dev/null +++ b/lib/seed/interfaces/response.ts @@ -0,0 +1,28 @@ +import { Field, ObjectType } from 'type-graphql'; + +export interface ErrorResponse { + statusCode: 400 | 403 | 404 | 500; + body: string; + [k: string]: any; +} + +@ObjectType() +export class SuccessResponse { + @Field() + success: boolean; +} + +export interface FunctionResponse { + success: boolean; + message: string; + data: T; +} + +export const fError = (message: string, data?: any): FunctionResponse => { + console.error(message, data); + return { success: false, message: message, data }; +}; + +export const fSuccess = (data?: any, message?: string): FunctionResponse => { + return { success: true, message: message || 'success', data }; +}; diff --git a/lib/seed/services/auth/FirebaseAuthService.ts b/lib/seed/services/auth/FirebaseAuthService.ts new file mode 100644 index 0000000..64c1b9c --- /dev/null +++ b/lib/seed/services/auth/FirebaseAuthService.ts @@ -0,0 +1,147 @@ +/* eslint-disable @typescript-eslint/camelcase */ +import { apiCall } from '@seed/helpers/FetchHelper'; +import { FirebaseTokenResult } from '@seed/interfaces/components'; +import { ApolloError } from 'apollo-server-lambda'; +import Firebase from './FirebaseService'; +import { LoginInput, LinkEmailInput, SMSVerif, CodeConfirmInput } from '@services/module-accounts/account.components'; + +import { v4 as uuid } from 'uuid'; +import { newError } from '@seed/helpers/Error'; + +const firebaseAuthUrl = 'https://identitytoolkit.googleapis.com/v1'; + +import { google, identitytoolkit_v3 } from 'googleapis'; +import { SuccessResponse } from '@seed/interfaces/response'; + +export default class FirebaseAuth { + private static instance: FirebaseAuth; + private apiKey: string; + + private identityToolkit: identitytoolkit_v3.Identitytoolkit; + + private constructor() { + this.apiKey = process.env.FIREBASE_apiKey || ''; + this.identityToolkit = google.identitytoolkit({ + auth: this.apiKey || '', + version: 'v3', + }); + } + + public static getInstance(): FirebaseAuth { + if (!FirebaseAuth.instance) { + FirebaseAuth.instance = new FirebaseAuth(); + } + + return FirebaseAuth.instance; + } + /* + * Login functions + */ + + public async loginWithEmailPassword({ email, password }: LoginInput): Promise { + try { + const requestUrl = `${firebaseAuthUrl}/accounts:signInWithPassword?key=${this.apiKey}`; + const res = await apiCall(requestUrl, 'POST', null, { + email, + password, + returnSecureToken: true, + }); + return res; + } catch (err) { + throw newError(2010, err); + } + } + + public async loginWithRefreshToken(refreshToken: string): Promise { + try { + const requestUrl = `${firebaseAuthUrl}/token?key=${this.apiKey}`; + const res = await apiCall(requestUrl, 'POST', null, { + grant_type: 'refresh_token', + refresh_token: refreshToken, + }); + + return { + localId: res.user_id, + idToken: res.id_token, + refreshToken: res.refresh_token, + expiresIn: res.expires_in, + }; + } catch (err) { + // throw newError('auth.error.' + err.error.error.message, err.error.error.code, err); + throw newError(2010, err); + } + } + + public async loginAnonymous(): Promise { + try { + const requestUrl = `${firebaseAuthUrl}/accounts:signUp?key=${this.apiKey}`; + const res = await apiCall(requestUrl, 'POST', null, { + returnSecureToken: true, + }); + + return res; + } catch (err) { + // throw newError('auth.error.' + err.error.error.message, err.error.error.code, err); + throw newError(2010, err); + } + } + + public async linkWithEmailPassword({ idToken, email, password }: LinkEmailInput): Promise { + try { + const requestUrl = `${firebaseAuthUrl}/accounts:update?key=${this.apiKey}`; + const res = await apiCall(requestUrl, 'POST', null, { + idToken, + email, + password, + returnSecureToken: true, + }); + return res; + } catch (err) { + throw newError(2015, err); + } + } + + /* + * MFA SMS functions + */ + + public async sendSMSVerification({ phoneNumber, recaptchaToken }: SMSVerif): Promise { + const response = new SuccessResponse(); + + const gRes = await this.identityToolkit.relyingparty.sendVerificationCode({ + requestBody: { + phoneNumber, + recaptchaToken, + }, + }); + + // save sessionInfo into db. You will need this to verify the SMS code + const sessionInfo = gRes.data.sessionInfo; + + console.log(sessionInfo); + + response.success = true; + + return response; + } + + public async checkSMSVerification({ email, code }: CodeConfirmInput): Promise { + const response = new SuccessResponse(); + + const sessionInfo = ''; + + const log = await this.identityToolkit.relyingparty.verifyPhoneNumber({ + requestBody: { + code: code.toString(), + phoneNumber: '', + sessionInfo, + }, + }); + + response.success = true; + + console.log(log); + + return response; + } +} diff --git a/lib/seed/services/auth/FirebaseService.ts b/lib/seed/services/auth/FirebaseService.ts new file mode 100644 index 0000000..e992977 --- /dev/null +++ b/lib/seed/services/auth/FirebaseService.ts @@ -0,0 +1,211 @@ +/* eslint-disable @typescript-eslint/camelcase */ +import FirebaseAdmin from 'firebase-admin'; +import axios from 'axios'; +import { newError } from '@seed/helpers/Error'; + +export interface FirebaseUser { + aud: string; + auth_time: number; + email: string | undefined; + email_verified: boolean | undefined; + exp: number; + firebase: { + identities: any; + sign_in_provider: 'custom' | 'password' | 'phone' | 'anonymous' | 'google.com' | 'facebook.com' | 'github.com' | 'twitter.com'; + }; + iat: number; + name: string; + picture: string | undefined; + uid: string; +} + +export default class Firebase { + private static instance: Firebase; + + private constructor() { + try { + const base = { + type: 'service_account', + project_id: process.env.FIREBASE_project_id || '', + private_key_id: process.env.FIREBASE_private_key_id || '', + private_key: process.env.FIREBASE_private_key ? process.env.FIREBASE_private_key.replace(/\\n/g, '\n') : '', + client_email: process.env.FIREBASE_client_email || '', + client_id: process.env.FIREBASE_client_id || '', + auth_uri: process.env.FIREBASE_auth_uri || '', + token_uri: process.env.FIREBASE_token_uri || '', + auth_provider_x509_cert_url: process.env.FIREBASE_auth_provider_x509_cert_url || '', + client_x509_cert_url: process.env.FIREBASE_client_x509_cert_url || '', + apiKey: process.env.FIREBASE_apiKey || '', + }; + + FirebaseAdmin.initializeApp({ + credential: FirebaseAdmin.credential.cert({ + projectId: base.project_id, + clientEmail: base.client_email, + privateKey: base.private_key, + }), + }); + } catch (error) { + if (!/already exists/.test(error.message)) { + console.error('Firebase initialization error raised', error.stack); + } + } + } + + public static getInstance(): Firebase { + if (!Firebase.instance) { + Firebase.instance = new Firebase(); + } + + return Firebase.instance; + } + /* + * IDENTIFY TOKEN + */ + + public async tokenIdentify(token: string): Promise { + try { + const res = await FirebaseAdmin.auth().verifyIdToken(token); + return res; + } catch (err) { + throw await newError(2011, err); + } + } + + /* + * ADMIN FUNCTIONS + */ + + public async createUser(obj: { email: string; password: string }): Promise { + try { + return await FirebaseAdmin.auth().createUser({ + ...obj, + emailVerified: true, + }); + } catch (err) { + throw newError(2012, err); + } + } + + public async getUserByEmail(email: string): Promise { + try { + return await FirebaseAdmin.auth().getUserByEmail(email); + } catch (err) { + throw newError(2013, err); + } + } + + public async updatePassword(obj: { uid: string; password: string }): Promise { + try { + return await FirebaseAdmin.auth().updateUser(obj.uid, { + password: obj.password, + emailVerified: true, + }); + } catch (err) { + throw newError(2013, err); + } + } + + public async updateEmail(obj: { uid: string; email: string }): Promise { + try { + return await FirebaseAdmin.auth().updateUser(obj.uid, { + email: obj.email, + emailVerified: true, + }); + } catch (err) { + throw newError(2013, err); + } + } + + public async deleteUser(obj: { uid: string }): Promise { + try { + return await FirebaseAdmin.auth().deleteUser(obj.uid); + } catch (err) { + throw newError(2013, err); + } + } + + public async getResetPasswordLink(email: string): Promise { + try { + return await FirebaseAdmin.auth().generatePasswordResetLink(email); + } catch (err) { + throw newError(2013, err); + } + } + + public async getEmailConfirmationLink(email: string): Promise { + try { + return await FirebaseAdmin.auth().generateEmailVerificationLink(email); + } catch (err) { + throw newError(2013, err); + } + } + + public async createTokenId(email: string): Promise { + try { + const firebaseUser = await FirebaseAdmin.auth().getUserByEmail(email); + + const customToken = await FirebaseAdmin.auth().createCustomToken(firebaseUser.uid); + + const r = await axios.post( + `https://www.googleapis.com/identitytoolkit/v3/relyingparty/verifyCustomToken?key=${process.env.FIREBASE_apiKey}`, + { + token: customToken, + returnSecureToken: true, + }, + ); + // console.log(r.data); + return r.data; + } catch (err) { + throw err; + } + } + + public async getMagicLink(email): Promise { + try { + const actionCodeSettings: any = { + // URL you want to redirect back to. The domain (www.example.com) for + // this URL must be whitelisted in the Firebase Console. + // This must be true for email link sign-in. + url: process.env.MAGIC_LINK_URL ? process.env.MAGIC_LINK_URL : 'https://makeit-studio.com/magicLink', + handleCodeInApp: true, + // FDL custom domain. + // dynamicLinkDomain: 'coolapp.page.link' + }; + + if (process.env.IOS_PACKAGE_NAME && process.env.IOS_PACKAGE_NAME != '') + actionCodeSettings.iOS = { + bundleId: process.env.IOS_PACKAGE_NAME, + }; + if (process.env.ANDROID_PACKAGE_NAME && process.env.ANDROID_PACKAGE_NAME != '') + actionCodeSettings.android = { + packageName: process.env.ANDROID_PACKAGE_NAME, + installApp: true, + }; + + return await FirebaseAdmin.auth().generateSignInWithEmailLink(email, actionCodeSettings); + } catch (err) { + throw err; + } + } + + /* + * PUSH NOTIFICATION FUNCTIONS + */ + + public async sendPushNotification(registrationToken: string | string[], title: string, body: string, data = {}): Promise { + try { + const firebaseInfo = await FirebaseAdmin.messaging().sendToDevice(registrationToken, { + data, + notification: { + title: title, + body: body, + }, + }); + console.log('firebase success', firebaseInfo); + } catch (error) { + console.log('firebase error', error); + } + return true; + } +} diff --git a/lib/seed/services/change-stream/change-stream.components.ts b/lib/seed/services/change-stream/change-stream.components.ts new file mode 100644 index 0000000..e15b33f --- /dev/null +++ b/lib/seed/services/change-stream/change-stream.components.ts @@ -0,0 +1,66 @@ +import { ObjectType, Field } from 'type-graphql'; + +import { registerEnumType } from 'type-graphql'; + +/* + ███████╗███╗ ██╗██╗ ██╗███╗ ███╗ + ██╔════╝████╗ ██║██║ ██║████╗ ████║ + █████╗ ██╔██╗ ██║██║ ██║██╔████╔██║ + ██╔══╝ ██║╚██╗██║██║ ██║██║╚██╔╝██║ + ███████╗██║ ╚████║╚██████╔╝██║ ╚═╝ ██║ + ╚══════╝╚═╝ ╚═══╝ ╚═════╝ ╚═╝ ╚═╝ +*/ + +export enum StreamOperationType { + insert = 'insert', + delete = 'delete', + replace = 'replace', + update = 'update', + hook = 'hook', +} +registerEnumType(StreamOperationType, { + name: 'StreamOperationType', +}); + +export enum PostHookStatus { + new = 'new', + inProcess = 'inProcess', + completed = 'completed', + error = 'error', + noAction = 'noAction', +} +registerEnumType(PostHookStatus, { + name: 'PostHookStatus', +}); + +/* + ██████╗ ██████╗ ███╗ ███╗██████╗ ██████╗ ███╗ ██╗███████╗███╗ ██╗████████╗███████╗ + ██╔════╝██╔═══██╗████╗ ████║██╔══██╗██╔═══██╗████╗ ██║██╔════╝████╗ ██║╚══██╔══╝██╔════╝ + ██║ ██║ ██║██╔████╔██║██████╔╝██║ ██║██╔██╗ ██║█████╗ ██╔██╗ ██║ ██║ ███████╗ + ██║ ██║ ██║██║╚██╔╝██║██╔═══╝ ██║ ██║██║╚██╗██║██╔══╝ ██║╚██╗██║ ██║ ╚════██║ + ╚██████╗╚██████╔╝██║ ╚═╝ ██║██║ ╚██████╔╝██║ ╚████║███████╗██║ ╚████║ ██║ ███████║ + ╚═════╝ ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═════╝ ╚═╝ ╚═══╝╚══════╝╚═╝ ╚═══╝ ╚═╝ ╚══════╝ +*/ + +/* + * AUTH + * + */ + +// @ObjectType() +// export class PostHookStatus { +// @Field() +// localId: string; +// @Field({ nullable: true }) +// email?: string; +// @Field({ nullable: true }) +// displayName?: string; +// @Field() +// idToken: string; +// @Field({ nullable: true }) +// registered?: boolean; +// @Field() +// refreshToken: string; +// @Field() +// expiresIn: string; +// } diff --git a/lib/seed/services/change-stream/change-stream.model.ts b/lib/seed/services/change-stream/change-stream.model.ts new file mode 100644 index 0000000..3906923 --- /dev/null +++ b/lib/seed/services/change-stream/change-stream.model.ts @@ -0,0 +1,68 @@ +import { ObjectType, Field, ArgsType } from 'type-graphql'; + +import { BaseGraphModel } from '@seed/graphql/BaseGraphModel'; +import { Permission } from '@seed/interfaces/permission'; + +import { GetArgs, GetManyArgs } from '@seed/graphql/Request'; + +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { StreamOperationType, PostHookStatus } from './change-stream.components'; +import GraphQLJSON from 'graphql-type-json'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.admin], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class ChangeStreamModel extends BaseGraphModel { + public constructor() { + super({ + //!!\\ Hard coded, do not change ! + collectionName: 'stream.changes', + permissions: permissions, + }); + } + + @Field(() => StreamOperationType) + operation: StreamOperationType; + + @Field() + collection: string; + + @Field() + documentKey: string; + + @Field(() => PostHookStatus) + hookStatus: PostHookStatus; + + @Field({ nullable: true }) + logMessage?: string; + + @Field(() => GraphQLJSON, { nullable: true }) + insertedValues?: Partial; + @Field(() => GraphQLJSON, { nullable: true }) + updatedValues?: Partial; + @Field(() => GraphQLJSON, { nullable: true }) + beforeUpdateValues?: Partial; + + logInfo?: any; + + searchOptions(): string[] { + return ['documentKey', 'hookStatus']; + } + filterOptions(): string[] { + return ['documentKey', 'hookStatus']; + } +} + +@ArgsType() +export class ChangeStreamArgs extends GetManyArgs { + @Field({ nullable: true }) + documentKey?: string; + + @Field({ nullable: true }) + hookStatus?: string; +} diff --git a/lib/seed/services/change-stream/change-stream.resolver.ts b/lib/seed/services/change-stream/change-stream.resolver.ts new file mode 100644 index 0000000..7b35b6a --- /dev/null +++ b/lib/seed/services/change-stream/change-stream.resolver.ts @@ -0,0 +1,27 @@ +import { Resolver } from 'type-graphql'; +import { ChangeStreamArgs } from './change-stream.model'; +import ChangeStreamModel from './change-stream.model'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { createBaseReadOnlyResolver } from '@seed/graphql/baseResolvers/BasePublicResolver'; + +const ChangeStreamBaseResolver = createBaseReadOnlyResolver('changeStreams', ChangeStreamModel, ChangeStreamArgs, [AccountTypeEnum.admin]); + +@Resolver(ChangeStreamModel) +export default class ChangeStreamAdminResolver extends ChangeStreamBaseResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ +} diff --git a/lib/seed/services/change-stream/change-stream.service.ts b/lib/seed/services/change-stream/change-stream.service.ts new file mode 100644 index 0000000..e69de29 diff --git a/lib/seed/services/database/DBRequestService.ts b/lib/seed/services/database/DBRequestService.ts new file mode 100644 index 0000000..bf2c187 --- /dev/null +++ b/lib/seed/services/database/DBRequestService.ts @@ -0,0 +1,561 @@ +import { GetArgs } from '@lib/seed/graphql/Request'; +import { parsePaginationOptions } from '@lib/seed/helpers/Request'; +import { ApolloContext } from '@lib/seed/interfaces/context'; +import { EngineModel } from '@seed/engine/EngineModel'; +import { BaseGraphModel } from '@seed/graphql/BaseGraphModel'; +import { newError } from '@seed/helpers/Error'; +import { escapeRegex } from '@seed/helpers/StringHelper'; +import { DateTabType, RangeType } from '@seed/interfaces/components'; +import { GeolocSearchComponent } from '@seed/interfaces/components.geo'; +import { getGeoLocFromAddress } from '@seed/services/geolocation/services/GooglePlaceService'; +import { ApolloError } from 'apollo-server-lambda'; +import moment from 'moment'; +import { AggregationCursor, Collection } from 'mongodb'; + +type searchType = 'dateTab' | 'dateRange' | 'hourRange' | 'beetween' | 'bool' | 'regex' | 'nameRegex'; +interface SearchEngineValue { + operation: + | 'dateTab' + | 'dateRange' + | 'hourRange' + | 'between' + | '$eq' + | '$gt' + | '$gte' + | '$in' + | '$lt' + | '$lte' + | '$ne' + | '$nin' + | '$not' + | '$exists' + | 'bool' + | 'regex' + | 'nameRegex' + | 'freeInput' + | 'searchText'; + dateTabConfig?: { + dbFromField: string; + dbToField: string; + }; + dateRangeConfig?: { + inputFromField: keyof T; + inputToField: keyof T; + dbFromField: string; + dbToField: string; + dateRangeType?: RangeType; + }; + hourRangeConfig?: { + inputFromField: keyof T; + inputToField: keyof T; + dbMainField: string; + dbSubfieldFrom: string; + dbSubfieldTo: string; + dateRangeType?: RangeType; + }; + beetweenConfig?: { + inputFromField: keyof T; + inputToField: keyof T; + }; + freeInputConfig?: { + searchOptions: string[]; + }; + dbFName?: string; + forcedValue?: any; +} +export type CustomSearchEngine = { + [P in keyof T]: SearchEngineValue; +} & + { + [P in searchType]?: SearchEngineValue; + } & { + [key: string]: SearchEngineValue; + }; + +export const createSearchRequest = (model: T, args: { search?: string }): any[] | undefined => { + if (!args.search) return; + + const search = args.search; + const searchValue = escapeRegex(search); + + if (model.searchOptions) { + const searchOptions = model.searchOptions(); + if (searchOptions.length > 0) { + const $or: any[] = []; + for (let index = 0; index < searchOptions.length; index++) { + const element = searchOptions[index]; + $or.push({ [`${element}`]: { $regex: `.*${searchValue}.*`, $options: 'i' } }); + } + return $or; + } + } + + return; +}; +export const addSearchRequest = (model: T, args: { search?: string }, filters: { $and: any[] }): any | undefined => { + const $or = createSearchRequest(model, args); + if ($or) filters.$and.push({ $or }); +}; + +export const createDateRangeRequest = (fromField: string, toField: string, args: any, type: RangeType = RangeType.intersect): any | undefined => { + try { + if ((args[fromField] && !args[toField]) || (args[toField] && !args[fromField])) throw newError(3005, '403'); + if (args[fromField] && args[toField]) { + if (type == RangeType.strict) + return { [`${fromField}`]: { $gte: moment(args[fromField]).toDate() }, [`${toField}`]: { $lte: moment(args[toField]).toDate() } }; + else if (type == RangeType.intersect) + return { + $or: [ + { [`${fromField}`]: { $gte: moment(args[fromField]).toDate(), $lte: moment(args[toField]).toDate() } }, + { [`${toField}`]: { $gte: moment(args[fromField]).toDate(), $lte: moment(args[toField]).toDate() } }, + ], + }; + else + return { + $or: [ + { [`${fromField}`]: { $gte: moment(args[fromField]).toDate() } }, + { [`${toField}`]: { $lte: moment(args[toField]).toDate() } }, + ], + }; + } + + return; + } catch (error) { + throw error; + } +}; +export const addDateRangeRequest = ( + fromField: string, + toField: string, + args: any, + filters: { $and: any[] }, + type: RangeType = RangeType.intersect, +): any | undefined => { + try { + const dateQuery = createDateRangeRequest(fromField, toField, args, type); + if (dateQuery) filters.$and.push(dateQuery); + } catch (error) { + throw error; + } +}; + +export const createTabRangeRequest = (fromField: string, toField: string, type: DateTabType = DateTabType.today): any | undefined => { + let from; + let to; + switch (type) { + case DateTabType.today: + from = moment().startOf('day'); + to = moment().endOf('day'); + break; + case DateTabType.tomorrow: + from = moment() + .add(1, 'day') + .startOf('day'); + to = moment() + .add(1, 'day') + .endOf('day'); + break; + case DateTabType.upcomming: + from = moment() + .add(2, 'day') + .startOf('day'); + break; + case DateTabType.past: + to = moment().startOf('day'); + break; + default: + break; + } + + if (from && to) { + return { + $or: [{ [`${fromField}`]: { $gte: from.toDate(), $lte: to.toDate() } }, { [`${toField}`]: { $gte: from.toDate(), $lte: to.toDate() } }], + }; + } else if (from) { + return { + $or: [{ [`${fromField}`]: { $gte: from.toDate() } }, { [`${toField}`]: { $gte: from.toDate() } }], + }; + } else if (to) { + return { + $or: [{ [`${fromField}`]: { $lte: to.toDate() } }, { [`${toField}`]: { $lte: to.toDate() } }], + }; + } else return; +}; +export const addDateTabRequest = ( + fromField: string, + toField: string, + filters: { $and: any[] }, + type: DateTabType = DateTabType.today, +): any | undefined => { + try { + const dateQuery = createTabRangeRequest(fromField, toField, type); + if (dateQuery) filters.$and.push(dateQuery); + } catch (error) { + throw error; + } +}; + +export const addGeoAggregatePipeline = (longitude: number, latitude: number, radiusInMeter: number, query?: any): any | undefined => { + if (query) + return { + $geoNear: { + near: { type: 'Point', coordinates: [longitude, latitude] }, + distanceField: 'dist.calculated', + maxDistance: radiusInMeter, + query: query, + includeLocs: 'dist.location', + spherical: true, + }, + }; + + return { + $geoNear: { + near: { type: 'Point', coordinates: [longitude, latitude] }, + distanceField: 'dist.calculated', + maxDistance: radiusInMeter, + includeLocs: 'dist.location', + spherical: true, + }, + }; +}; + +export const createGeoAggreatePipeline = (input: { + collection: Collection; + longitude: number; + latitude: number; + radiusInMeter: number; + count?: boolean; + query?: any; + pagination?: { limit: number; skip: number }; +}): AggregationCursor => { + // eslint-disable-next-line prefer-const + let { collection, longitude, latitude, radiusInMeter, count, query, pagination } = input; + if (!pagination) pagination = { limit: 100, skip: 0 }; + + const pipe = addGeoAggregatePipeline(longitude, latitude, radiusInMeter, query); + if (count) { + return collection.aggregate([{ $geoNear: pipe.$geoNear }, { $count: 'count' }]); + } else { + return collection.aggregate([{ $geoNear: pipe.$geoNear }, { $skip: pagination.skip }, { $limit: pagination.limit }]); + } +}; + +export const addFilterRequest = (model: T, args: any, filters: { $and: any[] }, check = true): any | undefined => { + if (model.searchEngine) { + const searchEngine = model.searchEngine(); + for (const key in searchEngine) { + const el = searchEngine[key] as SearchEngineValue; + + const fieldName = el.dbFName || key; + + if (el.operation == 'between') { + if (!el.beetweenConfig) throw 'Bad search engine config'; + if (args[el.beetweenConfig.inputFromField] && args[el.beetweenConfig.inputToField]) + filters.$and.push({ + [`${fieldName}`]: { $gte: args[el.beetweenConfig.inputFromField], $lte: args[el.beetweenConfig.inputToField] }, + }); + else if (args[el.beetweenConfig.inputFromField]) + filters.$and.push({ [`${fieldName}`]: { $gte: args[el.beetweenConfig.inputFromField] } }); + else if (args[el.beetweenConfig.inputToField]) + filters.$and.push({ [`${fieldName}`]: { $lte: args[el.beetweenConfig.inputToField] } }); + } else if (el.operation == 'dateRange') { + if (!el.dateRangeConfig) throw 'Bad search engine config'; + + let { inputFromField, inputToField, dbFromField, dbToField, dateRangeType } = el.dateRangeConfig; + + // Check who has the args + const argsToUse = args[key] || args; + + if (argsToUse['rangeType']) dateRangeType = argsToUse['rangeType']; + + if (argsToUse[inputFromField] && argsToUse[inputToField]) { + if (dateRangeType == RangeType.strict) + if (dbFromField == dbToField) + filters.$and.push({ + [`${dbFromField}`]: { + $gte: moment(argsToUse[inputFromField]).toDate(), + $lte: moment(argsToUse[inputToField]).toDate(), + }, + }); + else + filters.$and.push({ + [`${dbFromField}`]: { $gte: moment(argsToUse[inputFromField]).toDate() }, + [`${dbToField}`]: { $lte: moment(argsToUse[inputToField]).toDate() }, + }); + else if (dateRangeType == RangeType.intersect) + filters.$and.push({ + $or: [ + { + [`${dbFromField}`]: { + $gte: moment(argsToUse[inputFromField]).toDate(), + $lte: moment(argsToUse[inputToField]).toDate(), + }, + }, + { + [`${dbToField}`]: { + $gte: moment(argsToUse[inputFromField]).toDate(), + $lte: moment(argsToUse[inputToField]).toDate(), + }, + }, + ], + }); + else if (dateRangeType == RangeType.included) { + if (dbFromField == dbToField) + filters.$and.push({ + [`${dbFromField}`]: { + $lte: moment(argsToUse[inputToField]).toDate(), + $gte: moment(argsToUse[inputFromField]).toDate(), + }, + }); + else + filters.$and.push({ + [`${dbFromField}`]: { $lte: moment(argsToUse[inputFromField]).toDate() }, + [`${dbToField}`]: { $gte: moment(argsToUse[inputToField]).toDate() }, + }); + } else + filters.$and.push({ + $or: [ + { [`${dbFromField}`]: { $gte: moment(argsToUse[inputFromField]).toDate() } }, + { [`${dbToField}`]: { $lte: moment(argsToUse[inputToField]).toDate() } }, + ], + }); + } + } else if (el.operation == 'hourRange') { + if (!el.hourRangeConfig) throw 'Bad search engine config'; + + const { inputFromField, inputToField, dbMainField, dbSubfieldFrom, dbSubfieldTo, dateRangeType } = el.hourRangeConfig; + + // Get the subfi + + if (args[inputFromField] && args[inputToField]) { + if (dateRangeType == RangeType.strict) + filters.$and.push({ + [`${dbMainField}`]: { + $elemMatch: { + [`${dbSubfieldFrom}`]: { $gte: args[inputFromField] }, + [`${dbSubfieldTo}`]: { $lte: args[inputToField] }, + }, + }, + }); + else if (dateRangeType == RangeType.intersect) + filters.$and.push({ + [`${dbMainField}`]: { + $elemMatch: { + $or: [ + { + [`${dbSubfieldFrom}`]: { + $gte: args[inputFromField], + $lte: args[inputToField], + }, + }, + { [`${dbSubfieldTo}`]: { $gte: args[inputFromField], $lte: args[inputToField] } }, + ], + }, + }, + }); + else if (dateRangeType == RangeType.included) + filters.$and.push({ + [`${dbMainField}`]: { + $elemMatch: { + [`${dbSubfieldFrom}`]: { $lte: args[inputFromField] }, + [`${dbSubfieldTo}`]: { $gte: args[inputToField] }, + }, + }, + }); + else + filters.$and.push({ + $elemMatch: { + $or: [ + { [`${dbSubfieldFrom}`]: { $gte: args[inputFromField] } }, + { [`${dbSubfieldTo}`]: { $lte: args[inputToField] } }, + ], + }, + }); + } + } else if (el.operation == 'dateTab') { + if (!el.dateTabConfig) throw 'Bad search engine config'; + const { dbFromField, dbToField } = el.dateTabConfig; + if (args.dateTabType) addDateTabRequest(dbFromField, dbToField, filters, args.dateTabType); + } else if (el.operation == 'bool') { + if (args[key] === false || args[key] === true) { + filters.$and.push({ [`${fieldName}`]: args[key] }); + } + } else if (el.operation == 'nameRegex') { + if (args[key]) { + const searchValue = escapeRegex(args[key]); + filters.$and.push({ + $expr: { + $regexMatch: { + input: { $concat: ['$firstName', ' ', '$lastName'] }, + regex: `.*${searchValue}.*`, + options: 'i', + }, + }, + }); + } + } else if (el.operation == 'regex') { + if (args[key]) { + const searchValue = args[key].replace(/[-[\]{}()*+?.,\\^$|#\s]/g, '\\$&'); + filters.$and.push({ [`${fieldName}`]: { $regex: `.*${searchValue}.*`, $options: 'i' } }); + } + } else if (el.operation == 'freeInput') { + if (!el.freeInputConfig) throw 'Bad search engine config'; + const freeInputValue = args[key]; + + if (freeInputValue) { + const searchValue = escapeRegex(freeInputValue); + + const searchOptions = el.freeInputConfig.searchOptions; + if (searchOptions.length > 0) { + const $or: any[] = []; + for (let index = 0; index < searchOptions.length; index++) { + const element = searchOptions[index]; + $or.push({ [`${element}`]: { $regex: `.*${searchValue}.*`, $options: 'i' } }); + } + filters.$and.push({ $or }); + } + } + } else if (el.operation == 'searchText') { + if (args[key]) { + const searchTextValue = escapeRegex(args[key]); + // if (searchTextValue) filters.$and.push({ $text: { $search: searchTextValue } }); + if (searchTextValue) filters.$and.push({ searchT: { $regex: `.*${searchTextValue}.*`, $options: 'i' } }); + } + } else { + if (el.forcedValue != undefined) filters.$and.push({ [`${fieldName}`]: { [`${el.operation}`]: el.forcedValue } }); + else if (args[key] != undefined) filters.$and.push({ [`${fieldName}`]: { [`${el.operation}`]: args[key] } }); + } + } + } else { + const filterOptions = model.filterOptions ? model.filterOptions() : null; + delete args.search; + + for (const key in args) { + const element = args[key]; + + if (check) { + if (filterOptions && filterOptions.includes(key)) { + if (key == '_ids') { + filters.$and.push({ _id: { $in: (element as string).replace(' ', '').split(',') } }); + } else filters.$and.push({ [`${key}`]: element }); + } + } else { + if (key == '_ids') { + filters.$and.push({ _id: { $in: (element as string).replace(' ', '').split(',') } }); + } else filters.$and.push({ [`${key}`]: element }); + } + } + } + + return; +}; + +export const baseSearchFunction = async (input: { + model: T; + query?: any; + count?: boolean; + pagination?: GetArgs; + all?: boolean; + engine?: boolean; + ctx: ApolloContext | null; +}): Promise => { + const { model, query, pagination, all, ctx } = input; + + let finalSearch: any = { $and: [] }; + let gSearch: GeolocSearchComponent | undefined; + + let count = false; + + if (query) { + const { search, geoSearch, geoSearchWithAddress, /*geoPlaceSearch,*/ afterCreatedAt, afterUpdatedAt, _ids, n_ids, ...filterArgument } = query; + + if (input.count) count = true; + + if (search) { + const $or = createSearchRequest(model, query); + if ($or) finalSearch.$and.push({ $or }); + } + + if (geoSearch) gSearch = geoSearch; + else if (geoSearchWithAddress) { + const geoloc = await getGeoLocFromAddress(geoSearchWithAddress.formattedAddress); + gSearch = { + longitude: geoloc.loc.coordinates[0], + latitude: geoloc.loc.coordinates[1], + radius: geoSearchWithAddress.radius, + }; + } + + if (filterArgument) { + addFilterRequest(model, { search, ...filterArgument }, finalSearch, false); + } + + if (_ids) finalSearch.$and.push({ _id: { $in: _ids } }); + // eslint-disable-next-line @typescript-eslint/camelcase + if (n_ids) finalSearch.$and.push({ _id: { $nin: n_ids } }); + + if (afterCreatedAt) finalSearch.$and.push({ createdAt: { $gte: new Date(afterCreatedAt) } }); + if (afterUpdatedAt) finalSearch.$and.push({ updatedAt: { $gte: new Date(afterUpdatedAt) } }); + } + + // if (query.search) finalSearch.$and.push({ search: { $regex: `.*${query.search}.*`, $options: 'i' } }); + + // Check if exist in DB + if (finalSearch.$and.length == 0) finalSearch = {}; + + try { + if (gSearch) { + let pipelineCursor; + const finalQ = model.getQuery(finalSearch, ctx); + + if (count) { + pipelineCursor = createGeoAggreatePipeline({ + ...gSearch, + collection: await model.db(), + query: finalQ, + radiusInMeter: gSearch.radius, + count: true, + }); + } else { + const paginationOption = parsePaginationOptions(pagination); + pipelineCursor = createGeoAggreatePipeline({ + ...gSearch, + collection: await model.db(), + query: finalQ, + radiusInMeter: gSearch.radius, + pagination: paginationOption, + }); + } + + if (input.engine) { + const result = await pipelineCursor.toArray(); + if (count) return result[0] || 0; + else return ((model as unknown) as EngineModel).plainToClass(result); + } else { + const result = await pipelineCursor.toArray(); + if (count) return result[0] || 0; + else return result; + } + } else { + if (input.engine) { + if (count) { + console.log('counting'); + return await ((model as unknown) as EngineModel).getCount({ query: finalSearch, ctx }); + } else { + if (all) return await ((model as unknown) as EngineModel).getAll({ query: finalSearch, ctx }); + return await ((model as unknown) as EngineModel).getMany({ query: finalSearch, pagination, ctx }); + } + } else { + if (count) { + console.log('counting'); + return await model.getCount(finalSearch, ctx); + } else { + if (all) return await model.getAll(finalSearch, ctx); + return await model.getMany(finalSearch, pagination, ctx); + } + } + } + } catch (error) { + throw error; + } +}; diff --git a/lib/seed/services/database/DBService.ts b/lib/seed/services/database/DBService.ts new file mode 100644 index 0000000..73874d1 --- /dev/null +++ b/lib/seed/services/database/DBService.ts @@ -0,0 +1,61 @@ +import { MongoClient, Db } from 'mongodb'; +import _, { difference } from 'lodash'; +import { differenceBetweenObject } from '@seed/helpers/Utils'; + +export const createSetRequest = (field: string, data: object, oldObject?: any): Partial => { + let dataToChange; + if (oldObject) dataToChange = differenceBetweenObject(data, oldObject); + else dataToChange = data; + + const req: any = {}; + for (const key in dataToChange) { + if (dataToChange.hasOwnProperty(key)) { + if (_.isPlainObject(dataToChange[key])) { + if (field != '') req[`${field}.${key}`] = createSetRequest(key, dataToChange[key]); + else req[`${key}`] = createSetRequest('', dataToChange[key]); + } else { + if (field != '') req[`${field}.${key}`] = dataToChange[key]; + else req[`${key}`] = dataToChange[key]; + } + } + } + + return req; +}; + +export default class DB { + private static instance: DB; + public mongoClient!: MongoClient; + public db!: Db; + + // eslint-disable-next-line @typescript-eslint/no-empty-function + private constructor() {} + + public static async getInstance(): Promise { + if (!this.instance || !this.instance.mongoClient || !this.instance.mongoClient.isConnected) { + this.instance = new DB(); + + let mongoUrl; + if (process.env.MONGO_USER && process.env.MONGO_PWD) + mongoUrl = `mongodb+srv://${process.env.MONGO_USER}:${process.env.MONGO_PWD}@${process.env.MONGO_URL}`; + else mongoUrl = process.env.MONGO_URL || ''; + const mongoDb = process.env.MONGO_DB || ''; + + try { + this.instance.mongoClient = await MongoClient.connect(mongoUrl, { + useNewUrlParser: true, + useUnifiedTopology: true, + // loggerLevel: 'debug', + }); + this.instance.db = this.instance.mongoClient.db(mongoDb); + if (process.env.NODE_ENV != 'production') console.log('[MongoDB] New Connect'); + } catch (error) { + throw error; + } + } else { + if (process.env.NODE_ENV != 'production') console.log('[MongoDB] Existing Connect'); + } + + return DB.instance; + } +} diff --git a/lib/seed/services/database/FirestoreService.ts b/lib/seed/services/database/FirestoreService.ts new file mode 100644 index 0000000..82d4454 --- /dev/null +++ b/lib/seed/services/database/FirestoreService.ts @@ -0,0 +1,67 @@ +/* eslint-disable @typescript-eslint/camelcase */ +import FirebaseAdmin from 'firebase-admin'; + +export interface FirebaseUser { + aud: string; + auth_time: number; + email: string | undefined; + email_verified: boolean | undefined; + exp: number; + firebase: { + identities: any; + sign_in_provider: 'custom' | 'password' | 'phone' | 'anonymous' | 'google.com' | 'facebook.com' | 'github.com' | 'twitter.com'; + }; + iat: number; + name: string; + picture: string | undefined; + uid: string; +} + +export default class Firestore { + private static instance: any; + public static db: FirebaseFirestore.Firestore; + + private constructor() { + try { + const base = { + type: 'service_account', + project_id: process.env.FIRESTORE_project_id || '', + private_key_id: process.env.FIRESTORE_private_key_id || '', + private_key: process.env.FIRESTORE_private_key ? process.env.FIRESTORE_private_key.replace(/\\n/g, '\n') : '', + client_email: process.env.FIRESTORE_client_email || '', + client_id: process.env.FIRESTORE_client_id || '', + auth_uri: process.env.FIRESTORE_auth_uri || '', + token_uri: process.env.FIRESTORE_token_uri || '', + auth_provider_x509_cert_url: process.env.FIRESTORE_auth_provider_x509_cert_url || '', + client_x509_cert_url: process.env.FIRESTORE_client_x509_cert_url || '', + apiKey: process.env.FIRESTORE_apiKey || '', + }; + + FirebaseAdmin.initializeApp( + { + credential: FirebaseAdmin.credential.cert({ + projectId: base.project_id, + clientEmail: base.client_email, + privateKey: base.private_key, + }), + }, + 'firestore', + ); + + Firestore.instance = FirebaseAdmin; + Firestore.db = FirebaseAdmin.firestore(); + } catch (error) { + if (!/already exists/.test(error.message)) { + console.error('Firebase initialization error raised', error.stack); + } + } + } + + public static getInstance(): FirebaseFirestore.Firestore { + if (!Firestore.instance) { + Firestore.instance = new Firestore(); + } + + return Firestore.db; + } +} diff --git a/lib/seed/services/database/LoaderService.ts b/lib/seed/services/database/LoaderService.ts new file mode 100644 index 0000000..cfaac0d --- /dev/null +++ b/lib/seed/services/database/LoaderService.ts @@ -0,0 +1,59 @@ +/* eslint-disable @typescript-eslint/explicit-function-return-type */ +import _ from 'lodash'; +import DL from 'dataloader'; +import { EngineModel } from '@seed/engine/EngineModel'; + +export class DataLoader extends DL { + async loadMany(keys: ArrayLike): Promise> { + const nkeys = _.without(keys, undefined, null) as any; + const res = await super.loadMany(nkeys); + return _.without(res, undefined, null) as any; + } +} + +/* V1 - Graph Model */ + +export const batchFunction = async (model: T, _ids: any, field: string) => { + const results = await (await (model as any).db()).find({ [`${field}`]: { $in: _ids } }).toArray(); + return _ids.map((_id) => results.find((result) => result[field] === _id)); +}; + +export const batchOneToManyFunction = async (model: T, _ids: any, field: string) => { + const results = await (await (model as any).db()).find({ [`${field}`]: { $in: _ids } }).toArray(); + return _ids.map((_id) => _.filter(results, { [`${field}`]: _id })); +}; + +export const batchOneToManyFilterFunction = async (model: T, _ids: any, field: string, filter: any) => { + const results = await (await (model as any).db()).find({ [`${field}`]: { $in: _ids }, ...filter }).toArray(); + return _ids.map((_id) => _.filter(results, { [`${field}`]: _id })); +}; + +export const buildLoader = (model: T, field = '_id') => { + return new DataLoader((keys) => { + return batchFunction(model, keys, field); + }); +}; + +export const buildOneToManyLoader = (model: T, field = '_id') => { + return new DataLoader>((keys) => { + return batchOneToManyFunction(model, keys, field); + }); +}; + +export const buildOneToManyFilterLoader = (model: T, field = '_id', filter: any) => { + return new DataLoader>((keys) => { + return batchOneToManyFilterFunction(model, keys, field, filter); + }); +}; + +/* V2 - Engine Model */ +export const batchEngineFunction = async >(model: T, _ids: any, field: string) => { + const results = await model.getAll({ query: { [`${field}`]: { $in: _ids } } }); + return _ids.map((_id) => results.find((result) => result[field] === _id)); +}; + +export const buildEngineLoader = , V = T>(model: T, field = '_id') => { + return new DataLoader((keys) => { + return batchEngineFunction(model, keys, field); + }); +}; diff --git a/lib/seed/services/email/EmailHelper.ts b/lib/seed/services/email/EmailHelper.ts new file mode 100644 index 0000000..1d003ca --- /dev/null +++ b/lib/seed/services/email/EmailHelper.ts @@ -0,0 +1,118 @@ +import { interpolate } from '@seed/helpers/Utils'; +import EmailSettingsModel from '@services/module-cms/functions/emails-settings/emails-settings.model'; +import { AvailableTranslation } from '@src/__components/components'; + +export const sendEmailsHelper = async ( + emailSettings: EmailSettingsModel, + finalLanguage: AvailableTranslation, + emails: string | string[], + data: any, +): Promise => { + const templateErrorReportingEmail = process.env.MJ_ERROR_EMAIL || 'software@makeit-studio.com'; + + const toEmails: any[] = []; + + if (!Array.isArray(emails)) emails = [emails]; + for (let index = 0; index < emails.length; index++) { + toEmails.push({ Email: emails[index] }); + } + + const ccEmails: any[] = []; + if (emailSettings.cci && emailSettings.cci.length > 0) + for (let index = 0; index < emailSettings.cci.length; index++) { + ccEmails.push({ Email: emailSettings.cci[index] }); + } + + const body = emailSettings.body && emailSettings.body[finalLanguage] ? emailSettings.body[finalLanguage] : ''; + const subject = emailSettings.subject && emailSettings.subject[finalLanguage] ? emailSettings.subject[finalLanguage] : ''; + const templateId = + emailSettings.templateId && emailSettings.templateId[finalLanguage] ? emailSettings.templateId[finalLanguage]!.replace(' ', '') : ''; + let messages; + if (!emailSettings.custom && emailSettings.subject && emailSettings.body) { + // Check if valide template + + // If no template -> Send only what there is in DB + if (!templateId || templateId == '') { + console.log('MJ.LOG', 'sending no template'); + + messages = [ + { + From: { + Email: emailSettings.fromEmail, + Name: emailSettings.fromName, + }, + To: toEmails, + Cc: ccEmails, + ReplyTo: { + Email: emailSettings.replyToEmail, + }, + HTMLPart: interpolate(body || '', data, { withEol: true }), + Subject: interpolate(subject || '', data, { withEol: true }), + }, + ]; + } + // If template -> Send with template ID + else { + console.log('MJ.LOG', 'sending template body'); + messages = [ + { + From: { + Email: emailSettings.fromEmail, + Name: emailSettings.fromName, + }, + To: toEmails, + Cc: ccEmails, + ReplyTo: { + Email: emailSettings.replyToEmail, + }, + TemplateID: parseInt(templateId), + TemplateLanguage: true, + TemplateErrorReporting: { + Email: templateErrorReportingEmail, + }, + Subject: interpolate(subject || '', data, { withEol: true }), + Variables: { + body: interpolate(body || '', data, { withEol: true }), + }, + }, + ]; + } + } else { + console.log('MJ.LOG', 'sending template data'); + + messages = [ + { + From: { + Email: emailSettings.fromEmail, + Name: emailSettings.fromName, + }, + To: toEmails, + Cc: ccEmails, + ReplyTo: { + Email: emailSettings.replyToEmail, + }, + TemplateID: parseInt(templateId), + TemplateLanguage: true, + TemplateErrorReporting: { + Email: templateErrorReportingEmail, + }, + Variables: data, + }, + ]; + } + + const mailjet = require('node-mailjet').connect(process.env.MJ_APIKEY_PUBLIC, process.env.MJ_APIKEY_PRIVATE); + try { + // eslint-disable-next-line @typescript-eslint/camelcase + const res = await mailjet.post('send', { version: 'v3.1', perform_api_call: true }).request({ + Messages: messages, + }); + console.log('MJ.SUCCESS', emailSettings.key, res.response.text); + + return true; + } catch (error) { + console.error('MJ.ERROR', emailSettings.key, error); + console.error('MJ MESSAGE', messages); + return false; + } +}; diff --git a/lib/seed/services/email/EmailService.ts b/lib/seed/services/email/EmailService.ts new file mode 100644 index 0000000..c8ceaaf --- /dev/null +++ b/lib/seed/services/email/EmailService.ts @@ -0,0 +1,20 @@ +import { AvailableTranslation } from '@src/__components/components'; +import EmailSettingsModel from '@services/module-cms/functions/emails-settings/emails-settings.model'; +import { sendEmailsHelper } from './EmailHelper'; + +const templateErrorReportingEmail = process.env.MJ_ERROR_EMAIL || 'software@makeit-studio.com'; + +export const sendEmail = async (key: string, language: AvailableTranslation | null, email: string[] | string, data: T): Promise => { + const emailSettings = new EmailSettingsModel(); + + const finalLanguage = language || (process.env.DEFAULT_LANGUAGE as any) || AvailableTranslation.en; + + try { + await emailSettings.getOne({ key }, null); + } catch (error) { + console.error('EMAIL.ERROR', { message: 'notfound', key }); + return false; + } + + return await sendEmailsHelper(emailSettings, finalLanguage as any, email, data); +}; diff --git a/lib/seed/services/geolocation/README.md b/lib/seed/services/geolocation/README.md new file mode 100644 index 0000000..6a92f2e --- /dev/null +++ b/lib/seed/services/geolocation/README.md @@ -0,0 +1,9 @@ +# Make-it Geoloc Module + +[![](https://ci6.googleusercontent.com/proxy/2HHmNs0zn0uGJb9RqML73pX6Bmd-BntMaIYR6IpXeIHKqn2_tG60C1pLNtMHoLHOk1I5CT8k5x9gvb-C1zuudHnD5HDOpvHljVYcKmDL--rb0Ar8IT4UB7YaG1c=s0-d-e1-ft#https://makeit-assets.s3.eu-central-1.amazonaws.com/signature_sanawar.png)](https://makeit-studio.com) + +![Build Status](https://travis-ci.org/joemccann/dillinger.svg?branch=master) + + + +npm install @googlemaps/google-maps-services-js geolib diff --git a/lib/seed/services/geolocation/__tests/geolocation.playground.gql b/lib/seed/services/geolocation/__tests/geolocation.playground.gql new file mode 100644 index 0000000..e69de29 diff --git a/lib/seed/services/geolocation/components/countries.ts b/lib/seed/services/geolocation/components/countries.ts new file mode 100644 index 0000000..eb9b4ee --- /dev/null +++ b/lib/seed/services/geolocation/components/countries.ts @@ -0,0 +1,514 @@ +import { registerEnumType } from 'type-graphql'; + +export enum CountryCodesComponentEnum { + 'AF' = 'AF', + 'AX' = 'AX', + 'AL' = 'AL', + 'DZ' = 'DZ', + 'AS' = 'AS', + 'AD' = 'AD', + 'AO' = 'AO', + 'AI' = 'AI', + 'AQ' = 'AQ', + 'AG' = 'AG', + 'AR' = 'AR', + 'AM' = 'AM', + 'AW' = 'AW', + 'AU' = 'AU', + 'AT' = 'AT', + 'AZ' = 'AZ', + 'BS' = 'BS', + 'BH' = 'BH', + 'BD' = 'BD', + 'BB' = 'BB', + 'BY' = 'BY', + 'BE' = 'BE', + 'BZ' = 'BZ', + 'BJ' = 'BJ', + 'BM' = 'BM', + 'BT' = 'BT', + 'BO' = 'BO', + 'BA' = 'BA', + 'BW' = 'BW', + 'BV' = 'BV', + 'BR' = 'BR', + 'IO' = 'IO', + 'BN' = 'BN', + 'BG' = 'BG', + 'BF' = 'BF', + 'BI' = 'BI', + 'KH' = 'KH', + 'CM' = 'CM', + 'CA' = 'CA', + 'CV' = 'CV', + 'KY' = 'KY', + 'CF' = 'CF', + 'TD' = 'TD', + 'CL' = 'CL', + 'CN' = 'CN', + 'CX' = 'CX', + 'CC' = 'CC', + 'CO' = 'CO', + 'KM' = 'KM', + 'CG' = 'CG', + 'CD' = 'CD', + 'CK' = 'CK', + 'CR' = 'CR', + 'CI' = 'CI', + 'HR' = 'HR', + 'CU' = 'CU', + 'CY' = 'CY', + 'CZ' = 'CZ', + 'DK' = 'DK', + 'DJ' = 'DJ', + 'DM' = 'DM', + 'DO' = 'DO', + 'EC' = 'EC', + 'EG' = 'EG', + 'SV' = 'SV', + 'GQ' = 'GQ', + 'ER' = 'ER', + 'EE' = 'EE', + 'ET' = 'ET', + 'FK' = 'FK', + 'FO' = 'FO', + 'FJ' = 'FJ', + 'FI' = 'FI', + 'FR' = 'FR', + 'GF' = 'GF', + 'PF' = 'PF', + 'TF' = 'TF', + 'GA' = 'GA', + 'GM' = 'GM', + 'GE' = 'GE', + 'DE' = 'DE', + 'GH' = 'GH', + 'GI' = 'GI', + 'GR' = 'GR', + 'GL' = 'GL', + 'GD' = 'GD', + 'GP' = 'GP', + 'GU' = 'GU', + 'GT' = 'GT', + 'GG' = 'GG', + 'GN' = 'GN', + 'GW' = 'GW', + 'GY' = 'GY', + 'HT' = 'HT', + 'HM' = 'HM', + 'VA' = 'VA', + 'HN' = 'HN', + 'HK' = 'HK', + 'HU' = 'HU', + 'IS' = 'IS', + 'IN' = 'IN', + 'ID' = 'ID', + 'IR' = 'IR', + 'IQ' = 'IQ', + 'IE' = 'IE', + 'IM' = 'IM', + 'IL' = 'IL', + 'IT' = 'IT', + 'JM' = 'JM', + 'JP' = 'JP', + 'JE' = 'JE', + 'JO' = 'JO', + 'KZ' = 'KZ', + 'KE' = 'KE', + 'KI' = 'KI', + 'KR' = 'KR', + 'KW' = 'KW', + 'KG' = 'KG', + 'LA' = 'LA', + 'LV' = 'LV', + 'LB' = 'LB', + 'LS' = 'LS', + 'LR' = 'LR', + 'LY' = 'LY', + 'LI' = 'LI', + 'LT' = 'LT', + 'LU' = 'LU', + 'MO' = 'MO', + 'MK' = 'MK', + 'MG' = 'MG', + 'MW' = 'MW', + 'MY' = 'MY', + 'MV' = 'MV', + 'ML' = 'ML', + 'MT' = 'MT', + 'MH' = 'MH', + 'MQ' = 'MQ', + 'MR' = 'MR', + 'MU' = 'MU', + 'YT' = 'YT', + 'MX' = 'MX', + 'FM' = 'FM', + 'MD' = 'MD', + 'MC' = 'MC', + 'MN' = 'MN', + 'ME' = 'ME', + 'MS' = 'MS', + 'MA' = 'MA', + 'MZ' = 'MZ', + 'MM' = 'MM', + 'NA' = 'NA', + 'NR' = 'NR', + 'NP' = 'NP', + 'NL' = 'NL', + 'AN' = 'AN', + 'NC' = 'NC', + 'NZ' = 'NZ', + 'NI' = 'NI', + 'NE' = 'NE', + 'NG' = 'NG', + 'NU' = 'NU', + 'NF' = 'NF', + 'MP' = 'MP', + 'NO' = 'NO', + 'OM' = 'OM', + 'PK' = 'PK', + 'PW' = 'PW', + 'PS' = 'PS', + 'PA' = 'PA', + 'PG' = 'PG', + 'PY' = 'PY', + 'PE' = 'PE', + 'PH' = 'PH', + 'PN' = 'PN', + 'PL' = 'PL', + 'PT' = 'PT', + 'PR' = 'PR', + 'QA' = 'QA', + 'RE' = 'RE', + 'RO' = 'RO', + 'RU' = 'RU', + 'RW' = 'RW', + 'BL' = 'BL', + 'SH' = 'SH', + 'KN' = 'KN', + 'LC' = 'LC', + 'MF' = 'MF', + 'PM' = 'PM', + 'VC' = 'VC', + 'WS' = 'WS', + 'SM' = 'SM', + 'ST' = 'ST', + 'SA' = 'SA', + 'SN' = 'SN', + 'RS' = 'RS', + 'SC' = 'SC', + 'SL' = 'SL', + 'SG' = 'SG', + 'SK' = 'SK', + 'SI' = 'SI', + 'SB' = 'SB', + 'SO' = 'SO', + 'ZA' = 'ZA', + 'GS' = 'GS', + 'ES' = 'ES', + 'LK' = 'LK', + 'SD' = 'SD', + 'SR' = 'SR', + 'SJ' = 'SJ', + 'SZ' = 'SZ', + 'SE' = 'SE', + 'CH' = 'CH', + 'SY' = 'SY', + 'TW' = 'TW', + 'TJ' = 'TJ', + 'TZ' = 'TZ', + 'TH' = 'TH', + 'TL' = 'TL', + 'TG' = 'TG', + 'TK' = 'TK', + 'TO' = 'TO', + 'TT' = 'TT', + 'TN' = 'TN', + 'TR' = 'TR', + 'TM' = 'TM', + 'TC' = 'TC', + 'TV' = 'TV', + 'UG' = 'UG', + 'UA' = 'UA', + 'AE' = 'AE', + 'GB' = 'GB', + 'US' = 'US', + 'UM' = 'UM', + 'UY' = 'UY', + 'UZ' = 'UZ', + 'VU' = 'VU', + 'VE' = 'VE', + 'VN' = 'VN', + 'VG' = 'VG', + 'VI' = 'VI', + 'WF' = 'WF', + 'EH' = 'EH', + 'YE' = 'YE', + 'ZM' = 'ZM', + 'ZW' = 'ZW', + 'XK' = 'XK', + 'KP' = 'KP', + 'SS' = 'SS', + 'SX' = 'SX', + 'BQ' = 'BQ', + 'CW' = 'CW', +} +registerEnumType(CountryCodesComponentEnum, { + name: 'CountryCodesComponentEnum', +}); + +export const isoCountries = { + AF: 'Afghanistan', + AX: 'Aland Islands', + AL: 'Albania', + DZ: 'Algeria', + AS: 'American Samoa', + AD: 'Andorra', + AO: 'Angola', + AI: 'Anguilla', + AQ: 'Antarctica', + AG: 'Antigua And Barbuda', + AR: 'Argentina', + AM: 'Armenia', + AW: 'Aruba', + AU: 'Australia', + AT: 'Austria', + AZ: 'Azerbaijan', + BS: 'Bahamas', + BH: 'Bahrain', + BD: 'Bangladesh', + BB: 'Barbados', + BY: 'Belarus', + BE: 'Belgium', + BZ: 'Belize', + BJ: 'Benin', + BM: 'Bermuda', + BT: 'Bhutan', + BO: 'Bolivia', + BA: 'Bosnia And Herzegovina', + BW: 'Botswana', + BV: 'Bouvet Island', + BR: 'Brazil', + IO: 'British Indian Ocean Territory', + BN: 'Brunei Darussalam', + BG: 'Bulgaria', + BF: 'Burkina Faso', + BI: 'Burundi', + KH: 'Cambodia', + CM: 'Cameroon', + CA: 'Canada', + CV: 'Cape Verde', + KY: 'Cayman Islands', + CF: 'Central African Republic', + TD: 'Chad', + CL: 'Chile', + CN: 'China', + CX: 'Christmas Island', + CC: 'Cocos (Keeling) Islands', + CO: 'Colombia', + KM: 'Comoros', + CG: 'Congo', + CD: 'Congo, Democratic Republic', + CK: 'Cook Islands', + CR: 'Costa Rica', + CI: "Cote D'Ivoire", + HR: 'Croatia', + CU: 'Cuba', + CY: 'Cyprus', + CZ: 'Czech Republic', + DK: 'Denmark', + DJ: 'Djibouti', + DM: 'Dominica', + DO: 'Dominican Republic', + EC: 'Ecuador', + EG: 'Egypt', + SV: 'El Salvador', + GQ: 'Equatorial Guinea', + ER: 'Eritrea', + EE: 'Estonia', + ET: 'Ethiopia', + FK: 'Falkland Islands (Malvinas)', + FO: 'Faroe Islands', + FJ: 'Fiji', + FI: 'Finland', + FR: 'France', + GF: 'French Guiana', + PF: 'French Polynesia', + TF: 'French Southern Territories', + GA: 'Gabon', + GM: 'Gambia', + GE: 'Georgia', + DE: 'Germany', + GH: 'Ghana', + GI: 'Gibraltar', + GR: 'Greece', + GL: 'Greenland', + GD: 'Grenada', + GP: 'Guadeloupe', + GU: 'Guam', + GT: 'Guatemala', + GG: 'Guernsey', + GN: 'Guinea', + GW: 'Guinea-Bissau', + GY: 'Guyana', + HT: 'Haiti', + HM: 'Heard Island & Mcdonald Islands', + VA: 'Holy See (Vatican City State)', + HN: 'Honduras', + HK: 'Hong Kong', + HU: 'Hungary', + IS: 'Iceland', + IN: 'India', + ID: 'Indonesia', + IR: 'Iran, Islamic Republic Of', + IQ: 'Iraq', + IE: 'Ireland', + IM: 'Isle Of Man', + IL: 'Israel', + IT: 'Italy', + JM: 'Jamaica', + JP: 'Japan', + JE: 'Jersey', + JO: 'Jordan', + KZ: 'Kazakhstan', + KE: 'Kenya', + KI: 'Kiribati', + KR: 'Korea', + KW: 'Kuwait', + KG: 'Kyrgyzstan', + LA: "Lao People's Democratic Republic", + LV: 'Latvia', + LB: 'Lebanon', + LS: 'Lesotho', + LR: 'Liberia', + LY: 'Libyan Arab Jamahiriya', + LI: 'Liechtenstein', + LT: 'Lithuania', + LU: 'Luxembourg', + MO: 'Macao', + MK: 'Macedonia', + MG: 'Madagascar', + MW: 'Malawi', + MY: 'Malaysia', + MV: 'Maldives', + ML: 'Mali', + MT: 'Malta', + MH: 'Marshall Islands', + MQ: 'Martinique', + MR: 'Mauritania', + MU: 'Mauritius', + YT: 'Mayotte', + MX: 'Mexico', + FM: 'Micronesia, Federated States Of', + MD: 'Moldova', + MC: 'Monaco', + MN: 'Mongolia', + ME: 'Montenegro', + MS: 'Montserrat', + MA: 'Morocco', + MZ: 'Mozambique', + MM: 'Myanmar', + NA: 'Namibia', + NR: 'Nauru', + NP: 'Nepal', + NL: 'Netherlands', + AN: 'Netherlands Antilles', + NC: 'New Caledonia', + NZ: 'New Zealand', + NI: 'Nicaragua', + NE: 'Niger', + NG: 'Nigeria', + NU: 'Niue', + NF: 'Norfolk Island', + MP: 'Northern Mariana Islands', + NO: 'Norway', + OM: 'Oman', + PK: 'Pakistan', + PW: 'Palau', + PS: 'Palestinian Territory, Occupied', + PA: 'Panama', + PG: 'Papua New Guinea', + PY: 'Paraguay', + PE: 'Peru', + PH: 'Philippines', + PN: 'Pitcairn', + PL: 'Poland', + PT: 'Portugal', + PR: 'Puerto Rico', + QA: 'Qatar', + RE: 'Reunion', + RO: 'Romania', + RU: 'Russian Federation', + RW: 'Rwanda', + BL: 'Saint Barthelemy', + SH: 'Saint Helena', + KN: 'Saint Kitts And Nevis', + LC: 'Saint Lucia', + MF: 'Saint Martin', + PM: 'Saint Pierre And Miquelon', + VC: 'Saint Vincent And Grenadines', + WS: 'Samoa', + SM: 'San Marino', + ST: 'Sao Tome And Principe', + SA: 'Saudi Arabia', + SN: 'Senegal', + RS: 'Serbia', + SC: 'Seychelles', + SL: 'Sierra Leone', + SG: 'Singapore', + SK: 'Slovakia', + SI: 'Slovenia', + SB: 'Solomon Islands', + SO: 'Somalia', + ZA: 'South Africa', + GS: 'South Georgia And Sandwich Isl.', + ES: 'Spain', + LK: 'Sri Lanka', + SD: 'Sudan', + SR: 'Suriname', + SJ: 'Svalbard And Jan Mayen', + SZ: 'Swaziland', + SE: 'Sweden', + CH: 'Switzerland', + SY: 'Syrian Arab Republic', + TW: 'Taiwan', + TJ: 'Tajikistan', + TZ: 'Tanzania', + TH: 'Thailand', + TL: 'Timor-Leste', + TG: 'Togo', + TK: 'Tokelau', + TO: 'Tonga', + TT: 'Trinidad And Tobago', + TN: 'Tunisia', + TR: 'Turkey', + TM: 'Turkmenistan', + TC: 'Turks And Caicos Islands', + TV: 'Tuvalu', + UG: 'Uganda', + UA: 'Ukraine', + AE: 'United Arab Emirates', + GB: 'United Kingdom', + US: 'United States', + UM: 'United States Outlying Islands', + UY: 'Uruguay', + UZ: 'Uzbekistan', + VU: 'Vanuatu', + VE: 'Venezuela', + VN: 'Viet Nam', + VG: 'Virgin Islands, British', + VI: 'Virgin Islands, U.S.', + WF: 'Wallis And Futuna', + EH: 'Western Sahara', + YE: 'Yemen', + ZM: 'Zambia', + ZW: 'Zimbabwe', +}; + +export const getCountryName = (countryCode: CountryCodesComponentEnum): string => { + if (isoCountries.hasOwnProperty(countryCode)) { + return isoCountries[countryCode]; + } else { + return countryCode; + } +}; diff --git a/lib/seed/services/geolocation/functions/GooglePlaceResolver.ts b/lib/seed/services/geolocation/functions/GooglePlaceResolver.ts new file mode 100644 index 0000000..df6af4d --- /dev/null +++ b/lib/seed/services/geolocation/functions/GooglePlaceResolver.ts @@ -0,0 +1,59 @@ +import { Resolver, Query, Arg, Mutation, Ctx, Args, Authorized, Field, ArgsType } from 'type-graphql'; + +import { ApolloContext } from '@seed/interfaces/context'; + +import { GraphQLJSONObject } from 'graphql-type-json'; +import { MinLength } from 'class-validator'; +import { AvailableTranslation } from '@src/__components/components'; +import { CountryCodesComponentEnum } from '../components/countries'; +import { autocompleteService, getGeoLocFromAddress, getPlaceDetail } from '../services/GooglePlaceService'; + +@ArgsType() +export class AutocompleteArgs { + @Field() + session: string; + + @Field() + @MinLength(5) + input: string; + + @Field(() => AvailableTranslation, { nullable: true }) + language?: AvailableTranslation; + @Field(() => [CountryCodesComponentEnum], { nullable: true }) + country?: CountryCodesComponentEnum[]; +} + +@ArgsType() +export class GeolocFromLocationIdInput { + @Field() + session: string; + @Field() + placeId: string; +} + +@Resolver() +export default class GooglePlaceResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + @Query(() => [GraphQLJSONObject]) + async placesAutocomplete(@Args() args: AutocompleteArgs, @Ctx() ctx: ApolloContext): Promise { + return await autocompleteService(args.session, args.input, args.language, args.country); + } + + @Query(() => GraphQLJSONObject) + async placesGeocode(@Args() args: GeolocFromLocationIdInput, @Ctx() ctx: ApolloContext): Promise { + return await getPlaceDetail(args.session, args.placeId); + } + + @Query(() => GraphQLJSONObject) + async placesGeocodeFromAddress(@Arg('address') address: string, @Ctx() ctx: ApolloContext): Promise { + return await getGeoLocFromAddress(address); + } +} diff --git a/lib/seed/services/geolocation/models/BaseGeoSeoModel.ts b/lib/seed/services/geolocation/models/BaseGeoSeoModel.ts new file mode 100644 index 0000000..abd4672 --- /dev/null +++ b/lib/seed/services/geolocation/models/BaseGeoSeoModel.ts @@ -0,0 +1,66 @@ +import { ObjectType, Field, ArgsType, InputType, Args } from 'type-graphql'; + +import { GetManyArgs } from '@seed/graphql/Request'; + + +import { Permission } from '@seed/interfaces/permission'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import BaseSEOModel, { EditBaseSEOInput, NewBaseSEOInput } from '@seed/graphql/baseModels/BaseSeoModel'; +import { getDistanceBetween } from '../services/GeolocService'; +import { GeolocDistComponent, GeolocSearchComponent, PlaceComponent } from '@seed/interfaces/components.geo'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.public], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class BaseGeoSEOModel extends BaseSEOModel { + public constructor(collectionName?: string, perm?: Permission) { + const cName = collectionName ? collectionName : 'articles'; + const permission = perm ? perm : permissions; + super(cName, permission); + } + + @Field(() => PlaceComponent) + place: PlaceComponent; + + @Field(() => GeolocDistComponent, { nullable: true }) + dist?: GeolocDistComponent; + + // eslint-disable-next-line @typescript-eslint/explicit-function-return-type + @Field(() => Number, { nullable: true }) + async getDistFromLocation(@Args() location: GeolocSearchComponent) { + return getDistanceBetween( + { + latitude: this.place.loc.coordinates[1], + longitude: this.place.loc.coordinates[0], + }, + location, + ); + } + + searchOptions(): string[] { + return []; + } + filterOptions(): string[] { + return []; + } +} + +@ArgsType() +export class BaseSEOArgs extends GetManyArgs {} + +@InputType() +export class NewGeoBaseSEOInput extends NewBaseSEOInput implements Partial { + @Field(() => PlaceComponent) + place: PlaceComponent; +} + +@InputType() +export class EditGeoBaseSEOInput extends EditBaseSEOInput implements Partial { + @Field(() => PlaceComponent, { nullable: true }) + place?: PlaceComponent; +} diff --git a/lib/seed/services/geolocation/models/BaseGeoSimpleModel.ts b/lib/seed/services/geolocation/models/BaseGeoSimpleModel.ts new file mode 100644 index 0000000..dffb17e --- /dev/null +++ b/lib/seed/services/geolocation/models/BaseGeoSimpleModel.ts @@ -0,0 +1,83 @@ +import { ObjectType, Field, ArgsType, InputType, Args } from 'type-graphql'; + +import { GeolocAddressSearchComponent, GeolocDistComponent, GeolocSearchComponent, PlaceComponent, PlaceComponentOptionnalWithAddress } from '@seed/interfaces/components.geo'; +import { GetManyArgs } from '@seed/graphql/Request'; + +import { Permission } from '@seed/interfaces/permission'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import BaseSEOSimpleModel, { EditBaseSEOSimpleInput, NewBaseSEOSimpleInput } from '@seed/graphql/baseModels/BaseSeoSimpleModel'; +import { getDistanceBetween } from '../services/GeolocService'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.public], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class BaseGeoSimpleSEOModel extends BaseSEOSimpleModel { + public constructor(collectionName?: string, perm?: Permission) { + const cName = collectionName ? collectionName : 'articles'; + const permission = perm ? perm : permissions; + super(cName, permission); + } + + @Field(() => PlaceComponent) + place: PlaceComponent; + + @Field(() => GeolocDistComponent, { nullable: true }) + dist?: GeolocDistComponent; + + // eslint-disable-next-line @typescript-eslint/explicit-function-return-type + @Field(() => Number, { nullable: true }) + async getDistFromLocation(@Args() location: GeolocSearchComponent) { + return getDistanceBetween( + { + latitude: this.place.loc.coordinates[1], + longitude: this.place.loc.coordinates[0], + }, + location, + ); + } + + searchOptions(): string[] { + return []; + } + filterOptions(): string[] { + return []; + } +} + +@ArgsType() +export class BaseGeoArgs extends GetManyArgs { + @Field(() => GeolocSearchComponent, { nullable: true }) + geoSearch?: GeolocSearchComponent; + + @Field(() => GeolocAddressSearchComponent, { nullable: true }) + geoSearchWithAddress?: GeolocAddressSearchComponent; +} + +@InputType() +export class NewGeoBaseSEOSimpleInput extends NewBaseSEOSimpleInput implements Partial { + @Field(() => PlaceComponent) + place: PlaceComponent; +} + +@InputType() +export class EditGeoBaseSEOSimpleInput extends EditBaseSEOSimpleInput implements Partial { + @Field(() => PlaceComponent, { nullable: true }) + place?: PlaceComponent; +} + +@InputType() +export class NewGeoWithAddressBaseSEOSimpleInput extends NewBaseSEOSimpleInput implements Partial { + @Field(() => PlaceComponentOptionnalWithAddress) + place: PlaceComponentOptionnalWithAddress; +} + +@InputType() +export class EditGeoWithAddressBaseSEOSimpleInput extends EditBaseSEOSimpleInput implements Partial { + @Field(() => PlaceComponentOptionnalWithAddress, { nullable: true }) + place?: PlaceComponentOptionnalWithAddress; +} diff --git a/lib/seed/services/geolocation/services/GeolocService.ts b/lib/seed/services/geolocation/services/GeolocService.ts new file mode 100644 index 0000000..6a3b32d --- /dev/null +++ b/lib/seed/services/geolocation/services/GeolocService.ts @@ -0,0 +1,9 @@ +/* eslint-disable @typescript-eslint/no-var-requires */ +import env from '@config/.env.json'; + +import { getDistance } from 'geolib'; + +export const getDistanceBetween = (from: { latitude: number; longitude: number }, to: { latitude: number; longitude: number }): any => { + const res = getDistance(from, to); + return res.toFixed(2); +}; diff --git a/lib/seed/services/geolocation/services/GooglePlaceService.ts b/lib/seed/services/geolocation/services/GooglePlaceService.ts new file mode 100644 index 0000000..07302e5 --- /dev/null +++ b/lib/seed/services/geolocation/services/GooglePlaceService.ts @@ -0,0 +1,173 @@ +/* eslint-disable @typescript-eslint/no-var-requires */ +import { Client, PlaceAutocompleteResult } from '@googlemaps/google-maps-services-js'; +import { AvailableTranslation } from '@src/__components/components'; + +import { CountryCodesComponentEnum } from '@seed/services/geolocation/components/countries'; +import _ from 'lodash'; +import { AddressStrictComponent, PlaceComponent } from '@seed/interfaces/components.geo'; + +const client = new Client({}); + +export const getGeoLoc = async (address: { + number?: string; + street?: string; + zip?: string; + city?: string; + country?: string; +}): Promise<{ loc: any; formattedAddress: string }> => { + let addString; + addString = address.street || ''; + addString += ' ' + (address.number || ''); + addString += ' ' + (address.zip || ''); + addString += ' ' + (address.city || ''); + addString += ' ' + (address.country || ''); + + try { + const res = await client.geocode({ + params: { + address: addString, + key: process.env.GOOGLE_MAP_apiKey || '', + }, + timeout: 1000, // milliseconds + }); + + if (res.status != 200) throw 'no_result'; + + const loc = { + type: 'Point', + coordinates: [res.data.results[0].geometry.location.lng, res.data.results[0].geometry.location.lng], + }; + + return { + loc: loc, + formattedAddress: res[0].formattedAddress, + }; + } catch (err) { + throw err; + } +}; + +export const getGeoLocFromAddress = async (address: string): Promise<{ loc: any; formattedAddress: string }> => { + try { + const res = await client.geocode({ + params: { + address: address, + key: process.env.GOOGLE_MAP_apiKey || '', + }, + timeout: 1000, // milliseconds + }); + + if (res.status != 200) throw 'no_result'; + + const loc = { + type: 'Point', + coordinates: [res.data.results[0].geometry.location.lng, res.data.results[0].geometry.location.lat], + }; + + return { + loc: loc, + formattedAddress: res.data.results[0].formatted_address, + }; + } catch (err) { + throw err; + } +}; + +export const getPlaceFromAddress = async (address: AddressStrictComponent): Promise => { + try { + const res = await client.geocode({ + params: { + address: address.returnFullAddress(), + key: process.env.GOOGLE_MAP_apiKey || '', + }, + timeout: 1000, // milliseconds + }); + + if (res.status != 200) throw 'no_result'; + + const loc = { + type: 'Point', + coordinates: [res.data.results[0].geometry.location.lng, res.data.results[0].geometry.location.lat], + }; + + return { + placeId: res.data.results[0].place_id, + address, + loc: loc, + formattedAddress: res.data.results[0].formatted_address, + }; + } catch (err) { + throw err; + } +}; + + +export const getPlaceDetail = async ( + session: string, + placeId: string, +): Promise<{ loc: any; lat: number; lng: number; formattedAddress: string; otherInfo: any }> => { + try { + const res = await client.placeDetails({ + params: { + // eslint-disable-next-line @typescript-eslint/camelcase + place_id: placeId, + sessiontoken: session, + fields: ['address_component', 'adr_address', 'business_status', 'formatted_address', 'geometry', 'icon', 'name', 'photo'], + key: process.env.GOOGLE_MAP_apiKey || '', + }, + timeout: 1000, // milliseconds + }); + + if (res.status != 200) throw 'no_result'; + + if (res.data.result.geometry) { + const loc = { + type: 'Point', + coordinates: [res.data.result.geometry.location.lng, res.data.result.geometry.location.lat], + }; + + return { + loc: loc, + lat: res.data.result.geometry.location.lat, + lng: res.data.result.geometry.location.lng, + formattedAddress: res.data.result.formatted_address as string, + otherInfo: res.data.result, + }; + } else throw 'no_result'; + } catch (err) { + throw err; + } +}; + +export const autocompleteService = async ( + session: string, + input: string, + language: AvailableTranslation = AvailableTranslation.en, + countries?: CountryCodesComponentEnum[], +): Promise => { + const params: any = { + key: process.env.GOOGLE_MAP_apiKey || '', + input, + language, + sessiontoken: session, + }; + + if (countries) { + params.components = _.map(countries, (country) => { + return 'country:' + country.toLowerCase(); + }); + } + + try { + const res = await client.placeAutocomplete({ + params: params, + timeout: 1000, // milliseconds + }); + + if (res.status != 200) throw 'no_result'; + + return res.data.predictions; + } catch (err) { + throw err; + } +}; diff --git a/lib/seed/services/hooks/hooks.decorator.ts b/lib/seed/services/hooks/hooks.decorator.ts new file mode 100644 index 0000000..73ad712 --- /dev/null +++ b/lib/seed/services/hooks/hooks.decorator.ts @@ -0,0 +1,126 @@ +/* eslint-disable prefer-rest-params */ +import { PostHookStatus, StreamOperationType } from '@seed/engine/utils/streams/schemas/stream.components'; +import { createAsyncStream } from '@seed/engine/utils/streams/stream.service'; +import AWS from 'aws-sdk'; +import { v4 as uuid } from 'uuid'; +import DB from '../database/DBService'; + +export function asyncHookDecorator() { + return function (target: TFunction): any { + for (const prop of Object.getOwnPropertyNames(target.prototype)) { + // console.log(prop); + const oldFunc: Function = target.prototype[prop]; + + const triggerName = prop; + + if (oldFunc instanceof Function) { + target.prototype[prop] = async function (): Promise { + if (process.env.NODE_ENV === 'local' || process.env.isHook == 'true') { + console.log('[NOW - HOOKS]', triggerName); + return oldFunc.apply(this, arguments); + } else { + const functionName = process.env.AWS_Hooks || process.env.AWS_compute || ''; + + if (functionName && functionName != '') { + const params = { + FunctionName: functionName, + InvocationType: 'Event', + Payload: JSON.stringify({ trigger: triggerName }), + }; + + await createAsyncStream(triggerName as never, arguments); + + AWS.config.region = process.env.awsRegion || 'eu-west-1'; + const lambda = new AWS.Lambda(); + + try { + await lambda.invoke(params).promise(); + console.log('[ASYNC - HOOKS] - INVOKING SENT', triggerName); + } catch (error) { + console.error('[ASYNC - HOOKS] - ERROR', error); + } + + return; + } + + return oldFunc.apply(this, arguments); + } + }; + } + } + }; +} + +export async function asyncHook(data: any): Promise { + if (process.env.NODE_ENV === 'local') { + console.log('[LOCAL] - HOOKS', data.operation); + return true; + } else { + const functionName = process.env.AWS_Hooks || process.env.AWS_compute || ''; + + if (functionName && functionName != '') { + const params = { + FunctionName: functionName, + InvocationType: 'Event', + Payload: JSON.stringify({ streamId: data._id }), + }; + + AWS.config.region = process.env.awsRegion || 'eu-west-1'; + const lambda = new AWS.Lambda(); + + try { + await lambda.invoke(params).promise(); + console.log('[ASYNC - HOOKS] - INVOKING SENT', data.operation, data.collection); + } catch (error) { + console.error('[ASYNC - HOOKS] - ERROR', error); + } + } + } + return false; +} + +export async function asyncHookAndStream(hookName: string, data: any): Promise { + if (process.env.NODE_ENV === 'local') { + console.log('[LOCAL] - HOOKS', hookName); + return true; + } else if (process.env.isHook == 'true') { + console.log('[ONLINE] - HOOKS', hookName); + return true; + } else { + const functionName = process.env.AWS_Hooks || process.env.AWS_compute || ''; + + if (functionName && functionName != '') { + const changeStreamId = uuid(); + const changeStreamData = { + _id: changeStreamId, + operation: StreamOperationType.hook, + collection: hookName, + documentKey: '', + insertedValues: data, + hookStatus: PostHookStatus.new, + createdAt: new Date(), + updateAt: new Date(), + }; + await (await DB.getInstance()).db.collection('stream.changes').insertOne(changeStreamData); + + const params = { + FunctionName: functionName, + InvocationType: 'Event', + Payload: JSON.stringify({ streamId: changeStreamId }), + }; + + AWS.config.region = process.env.awsRegion || 'eu-west-1'; + const lambda = new AWS.Lambda(); + + try { + await lambda.invoke(params).promise(); + console.log('[ASYNC - HOOKS] - INVOKING SENT', hookName, changeStreamData._id); + } catch (error) { + console.error('[ASYNC - HOOKS] - ERROR', error); + } + } else { + console.error('[ASYNC - HOOKS] - NO COMPUTE'); + } + } + return false; +} diff --git a/lib/seed/services/notifications/helpers/notification.helper.ts b/lib/seed/services/notifications/helpers/notification.helper.ts new file mode 100644 index 0000000..38bc458 --- /dev/null +++ b/lib/seed/services/notifications/helpers/notification.helper.ts @@ -0,0 +1,59 @@ +import { EngineModel } from '@seed/engine/EngineModel'; +import { AvailableTranslation } from '@src/__components/components'; +import _ from 'lodash'; +import { NotificationInput } from '../services/NotificationService'; +import { NotificationType, ReceiverData } from '../schemas/notifications.schema'; +import { IEngineSchema } from '@seed/engine/EngineSchema'; + +export const getNotificationsTypesFromInput = (input: NotificationInput): NotificationType[] => { + const types: NotificationType[] = []; + + if (input.emails) types.push(NotificationType.mail); + if (input.pushTokens) types.push(NotificationType.mail); + if (input.sms) types.push(NotificationType.sms); + + return types; +}; + +export const getReceiverDataFromLanguage = (input: NotificationInput): typeof receiverData => { + const receiverData: { + emails?: ReceiverData[]; + pushTokens?: ReceiverData[]; + sms?: ReceiverData[]; + } = {}; + + const finalLanguage = input.language || (process.env.DEFAULT_LANGUAGE as any) || AvailableTranslation.en; + + if (input.emails) { + if(!_.isArray(input.emails)) input.emails = [input.emails]; + + receiverData.emails = _.map(input.emails, (email) => { + return { + value: email, + language: finalLanguage, + }; + }); + } + if (input.pushTokens) { + if(!_.isArray(input.pushTokens)) input.pushTokens = [input.pushTokens]; + + receiverData.pushTokens = _.map(input.pushTokens, (pushTokens) => { + return { + value: pushTokens, + language: finalLanguage, + }; + }); + } + if (input.sms) { + if(!_.isArray(input.sms)) input.sms = [input.sms]; + + receiverData.sms = _.map(input.sms, (sms) => { + return { + value: sms, + language: finalLanguage, + }; + }); + } + + return receiverData; +}; diff --git a/lib/seed/services/notifications/helpers/slack.helper.ts b/lib/seed/services/notifications/helpers/slack.helper.ts new file mode 100644 index 0000000..bc6824e --- /dev/null +++ b/lib/seed/services/notifications/helpers/slack.helper.ts @@ -0,0 +1,60 @@ +import { fetchCall } from '@seed/helpers/FetchHelper'; +import { interpolate, promiseAll } from '@seed/helpers/Utils'; +import EmailSettingsModel from '@services/module-cms/functions/emails-settings/emails-settings.model'; +import { AvailableTranslation } from '@src/__components/components'; + +export const sendToSlack = async (data, webhookUrl) => { + return fetchCall({ + url: webhookUrl, + method: 'POST', + body: data, + raw: true, + headers: { 'Content-Type': 'application/json' }, + }); +}; + +export const sendSlackHelper = async (emailSettings: EmailSettingsModel, data: any): Promise => { + const slackSettings = emailSettings.slackSettings; + if (!slackSettings) { + console.error('No slack settings found'); + return false; + } + const webhooksUrls = slackSettings.webhookUrls; + const text = interpolate(slackSettings.text, data, { withEol: true }); + + const slackNotification: any = { + username: slackSettings.username || 'Project User', + text, + // eslint-disable-next-line @typescript-eslint/camelcase + icon_emoji: slackSettings.icon || ':rocket:', // User icon, you can also use custom icons here + }; + + if (slackSettings.actions) { + const actions: any[] = []; + slackSettings.actions.forEach((element) => { + const a: any = { + type: 'button', + text: element.text, + url: element.url, + }; + if (element.style) a.style = element.style; + actions.push(a); + }); + slackNotification.attachments = [ + { + actions, + }, + ]; + } + + const promises: Promise[] = []; + + webhooksUrls.forEach((element) => { + promises.push(sendToSlack(slackNotification, element)); + }); + + const promiseResults = await promiseAll(promises); + console.log(promiseResults); + + return true; +}; diff --git a/lib/seed/services/notifications/helpers/sms.helper.ts b/lib/seed/services/notifications/helpers/sms.helper.ts new file mode 100644 index 0000000..e58c92d --- /dev/null +++ b/lib/seed/services/notifications/helpers/sms.helper.ts @@ -0,0 +1,63 @@ +import { fetchCall } from '@seed/helpers/FetchHelper'; +import { interpolate, promiseAll } from '@seed/helpers/Utils'; +import EmailSettingsModel from '@services/module-cms/functions/emails-settings/emails-settings.model'; +import { AvailableTranslation } from '@src/__components/components'; + +export const sendSmsHelper = async ( + emailSettings: EmailSettingsModel, + finalLanguage: AvailableTranslation, + numbers: string | string[], + data: any, +): Promise => { + const smsSettings = emailSettings.smsSettings; + if (!smsSettings || !smsSettings.text[finalLanguage]) { + console.error('No sms settings found for this language', finalLanguage); + return false; + } + + //const toNumbers: any[] = []; + + if (!Array.isArray(numbers)) numbers = [numbers]; + /* for (let index = 0; index < numbers.length; index++) { + toNumbers.push({ Email: numbers[index] }); + } */ + + const body = interpolate(smsSettings.text[finalLanguage] || '', data); + + const promises: Promise[] = []; + // eslint-disable-next-line @typescript-eslint/no-var-requires + const twilio = require('twilio'); + + const client = new twilio(process.env.TWILIO_ACCOUNT_SID, process.env.TWILIO_AUTH_TOKEN); + + numbers.forEach((element) => { + const dataToSend = { + body: body, + from: process.env.TWILIO_FROM_NUMBER || '', + to: element, + }; + + console.log('SMS - Sending to', dataToSend); + + promises.push( + /* fetchCall({ + url: `https://api.twilio.com/2010-04-01/Accounts/${process.env.TWILIO_ACCOUNT_SID}/Messages.json`, + method: 'POST', + query: dataToSend, + headers: { + Authorization: 'Basic ' + Buffer.from(`${process.env.TWILIO_ACCOUNT_SID}:${process.env.TWILIO_AUTH_TOKEN}`).toString('base64'), + "Content-Type": 'x-www-form-urlencoded', + }, + }), */ + + client.messages.create({ + ...dataToSend, + }), + ); + }); + + const promiseResults = await promiseAll(promises); + console.log(promiseResults); + + return true; +}; diff --git a/lib/seed/services/notifications/notifications.model.ts b/lib/seed/services/notifications/notifications.model.ts new file mode 100644 index 0000000..e939328 --- /dev/null +++ b/lib/seed/services/notifications/notifications.model.ts @@ -0,0 +1,41 @@ +import { ObjectType, Field, ID, ArgsType, InputType, registerEnumType, Int } from 'type-graphql'; + +import { Permission } from '@seed/interfaces/permission'; + +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { EngineModel } from '@seed/engine/EngineModel'; +import { NotificationDBInterfaceSchema, NotificationDBSchema, NotificationSchema } from './schemas/notifications.schema'; +import { plainToClass } from 'class-transformer'; +import { IEngineSchema } from '@seed/engine/EngineSchema'; +import { sendNotificationHanlder } from './services/NotificationService'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.public], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class NotificationModel extends EngineModel { + public constructor(input?: NotificationDBInterfaceSchema & Partial) { + const dataInit = plainToClass(NotificationDBSchema, input || {}); + super({ + collectionName: 'notifications', + permissions: permissions, + dataInit, + }); + } + + plainToClass(plain: any): NotificationSchema | NotificationSchema[] { + return plainToClass(NotificationSchema, plain); + } + + async afterCreate(): Promise { + await sendNotificationHanlder(this); + } + + async afterUpdate(): Promise {} + + async afterDelete(): Promise {} +} diff --git a/lib/seed/services/notifications/schemas/notifications.schema.ts b/lib/seed/services/notifications/schemas/notifications.schema.ts new file mode 100644 index 0000000..ccb92cf --- /dev/null +++ b/lib/seed/services/notifications/schemas/notifications.schema.ts @@ -0,0 +1,82 @@ +import { NotificationEnum } from '@config/config'; +import { IEngineSchema, MetaBy, MetaPermissions } from '@seed/engine/EngineSchema'; +import { AvailableTranslation } from '@src/__components/components'; +import { GraphQLJSONObject } from 'graphql-type-json'; +import { InputType, ObjectType, Field, Int, registerEnumType } from 'type-graphql'; + +export enum NotificationType { + 'mail' = 'mail', + 'push' = 'push', + 'sms' = 'sms', +} + +registerEnumType(NotificationType, { + name: 'NotificationType', +}); + +export enum NotificationStatus { + 'toSend' = 'toSend', + 'sent' = 'sent', + 'error' = 'error', +} + +registerEnumType(NotificationStatus, { + name: 'NotificationStatus', +}); + +@ObjectType() +@InputType() +export class ReceiverData { + @Field() + value: string; + + @Field(() => AvailableTranslation) + language: AvailableTranslation; +} + +@ObjectType() +@InputType() +export class NotificationBaseSchema { + @Field() + ressourceId: string; + + @Field(() => [String]) + accountIds?: string[]; + + data: any; + + @Field(() => NotificationEnum) + key: NotificationEnum; + + @Field(() => [ReceiverData], { nullable: true }) + emails?: ReceiverData[]; + @Field(() => [ReceiverData], { nullable: true }) + pushTokens?: ReceiverData[]; + @Field(() => [ReceiverData], { nullable: true }) + sms?: ReceiverData[]; + + @Field({ nullable: true }) + slack?: boolean; + + @Field(() => NotificationType) + status: NotificationStatus; + + @Field({ nullable: true }) + sentAt?: Date; +} + +@ObjectType() +export class NotificationDBInterfaceSchema extends NotificationBaseSchema {} + +@ObjectType() +export class NotificationDBSchema extends NotificationDBInterfaceSchema {} + +@ObjectType({ implements: IEngineSchema }) +export class NotificationSchema extends NotificationDBSchema implements IEngineSchema { + _id: string; + organisationId?: string | undefined; + by?: MetaBy | undefined; + permissions: MetaPermissions; + createdAt: Date; + updatedAt: Date; +} diff --git a/lib/seed/services/notifications/services/NotificationService.ts b/lib/seed/services/notifications/services/NotificationService.ts new file mode 100644 index 0000000..8b878be --- /dev/null +++ b/lib/seed/services/notifications/services/NotificationService.ts @@ -0,0 +1,93 @@ +import { AvailableTranslation } from '@src/__components/components'; +import { promiseAll } from '@seed/helpers/Utils'; +import EmailSettingsModel from '@services/module-cms/functions/emails-settings/emails-settings.model'; +import { sendEmailsHelper } from '../../email/EmailHelper'; +import NotificationModel from '../notifications.model'; +import _ from 'lodash'; +import { NotificationDBInterfaceSchema, NotificationStatus } from '../schemas/notifications.schema'; +import { getReceiverDataFromLanguage } from '../helpers/notification.helper'; +import { IEngineSchema } from '@seed/engine/EngineSchema'; +import AccountModel from '@src/accounts/account.model'; +import { sendSmsHelper } from '../helpers/sms.helper'; +import { sendSlackHelper } from '../helpers/slack.helper'; +import { NotificationEnum } from '@config/config'; +import { EngineNotificationEnum } from '@seed/interfaces/components.notifications'; +import { SettingsCache } from '@seed/graphql/Settings'; + +export interface NotificationInput { + ressource: T | any; + key: NotificationEnum & EngineNotificationEnum | NotificationEnum; + emails?: string[] | string; + pushTokens?: string[] | string; + sms?: string[] | string; + slack?: boolean; + language?: AvailableTranslation; + account?: AccountModel; +} + +export const sendNotification = async (input: NotificationInput): Promise => { + const { ressource, key, account } = input; + + // Create the notification + const notificationData: NotificationDBInterfaceSchema = { + ressourceId: ressource._id || 'no-data', + data: ressource, + key, + ...getReceiverDataFromLanguage(input), + slack: input.slack, + status: NotificationStatus.toSend, + }; + + if (account) notificationData.accountIds = [account._id]; + + const notificationModel = new NotificationModel(notificationData); + + await notificationModel.saveOne({}); + + return true; +}; + +export const sendNotificationHanlder = async (input: NotificationModel): Promise => { + const { data, key, emails, sms, slack, accountIds } = input.get(); + + let emailSettings: EmailSettingsModel[]; + try { + emailSettings = SettingsCache.getInstance().cache.emailsContent[key]; + } catch (error) { + console.error('NOTIFICATION.ERROR', { message: 'notfound', key }); + return false; + } + + const promises: any[] = []; + + emailSettings.forEach(emailSetting => { + // Email flows + if (emails) { + // Send one per language + const emailsByLanguage = _.groupBy(emails, 'language'); + for (const language in emailsByLanguage) { + const emails = _.map(emailsByLanguage[language], 'value'); + promises.push(sendEmailsHelper(emailSetting, language as any, emails, data)); + } + } + if (sms) { + // Send one per language + const smsByLanguage = _.groupBy(sms, 'language'); + for (const language in smsByLanguage) { + const numbers = _.map(smsByLanguage[language], 'value'); + promises.push(sendSmsHelper(emailSetting, language as any, numbers, data)); + } + } + if (slack) { + promises.push(sendSlackHelper(emailSetting, data)); + } + }) + + + const promiseResults = await promiseAll(promises); + console.log(promiseResults); + + await input.updateOne({ query: { _id: input._id }, newData: { status: NotificationStatus.sent } }); + + return true; +}; diff --git a/lib/seed/webpack.config.js b/lib/seed/webpack.config.js new file mode 100644 index 0000000..1ffc9f1 --- /dev/null +++ b/lib/seed/webpack.config.js @@ -0,0 +1,50 @@ +/* eslint-disable @typescript-eslint/no-var-requires */ +const path = require('path'); +const nodeExternals = require('webpack-node-externals'); +const slsw = require('serverless-webpack'); +const ForkTsCheckerWebpackPlugin = require('fork-ts-checker-webpack-plugin'); +const TsConfigPathsPlugin = require('awesome-typescript-loader').TsConfigPathsPlugin; +const CopyPlugin = require("copy-webpack-plugin"); + + +const isLocal = slsw.lib.webpack.isLocal; +const stage = slsw.lib.options.stage; + +console.log('[Webpack - RUN]', new Date()); +console.log('[Webpack - RUN] Entry', slsw.lib.entries); +console.log('[Webpack - RUN] Stage', stage); + +module.exports = { + mode: isLocal ? 'development' : 'production', + entry: slsw.lib.entries, + externals: [nodeExternals({ + allowlist:["axios"], + })], + devtool: 'source-map', + resolve: { + extensions: ['.js', '.jsx', '.json', '.ts', '.tsx'], + plugins: [new TsConfigPathsPlugin()], + }, + output: { + libraryTarget: 'commonjs2', + path: path.join(__dirname, '.webpack'), + filename: '[name].js', + }, + target: 'node', + module: { + rules: [ + // all files with a `.ts` or `.tsx` extension will be handled by `ts-loader` + { test: /\.tsx?$/, loader: 'awesome-typescript-loader', options: { transpileOnly: true }, exclude: /node_modules/ }, + ], + }, + plugins: [new ForkTsCheckerWebpackPlugin(), + new CopyPlugin({ + patterns: [ + { from: "node_modules/axios", to: "node_modules/axios" }, + ], + }),], + optimization: { + minimize: false, // <---- disables uglify. + nodeEnv: stage, + }, +}; diff --git a/package.json b/package.json new file mode 100644 index 0000000..a04bb54 --- /dev/null +++ b/package.json @@ -0,0 +1,121 @@ +{ + "name": "workinflex", + "version": "1.0.0", + "description": "Work in Flex by Make it", + "devops": { + "awsProfile": "workinflex", + "awsRegion": "eu-central-1", + "services": { + "admin": [ + "./lib/seed/devops", + "serverless-admin.yaml", + "handler-admin.ts" + ], + "app": [ + "./lib/seed/devops", + "serverless-app.yaml", + "handler-app.ts" + ], + "hooks": [ + "./lib/seed/devops", + "serverless-hooks.yaml", + "handler-hooks.ts" + ], + "uploads": [ + "./services/api-uploads", + "serverless-upload.yaml" + ], + "messaging": [ + "./services/api-messaging", + "serverless-messaging.yaml" + ], + "stripe-hooks": [ + "./services/module-payments", + "serverless-payments.yaml", + "__handlers/stripeHookHandler.ts" + ], + "cronjobs": [ + "./lib/__cronjobs", + "serverless-cron.yaml", + "handler.ts" + ] + } + }, + "scripts": { + "dev:app": "ts-node-dev --respawn --inspect=9922 -r tsconfig-paths/register ./lib/seed/app.ts", + "dev:admin": "ts-node-dev --respawn --inspect=9922 -r tsconfig-paths/register ./lib/seed/app-admin.ts", + "dev:messaging": "ts-node-dev --respawn --inspect=9922 -r tsconfig-paths/register ./services/api-messaging/app.ts", + "deploy:dev": "node ./lib/seed/devops/deploy.js dev", + "deploy:staging": "node ./lib/seed/devops/deploy.js staging", + "deploy:production": "node ./lib/seed/devops/deploy.js production", + "deploy:domains": "node ./lib/seed/devops/deploy-domains.js", + "test:gen:app": "zeus http://localhost:4000 ./__tests/app/helpers --ts", + "test:gen:admin": "zeus http://localhost:4001 ./__tests/admin/helpers --ts", + "test:app": "ts-node-dev --respawn -r tsconfig-paths/register ./__tests/app", + "test:admin": "ts-node-dev --respawn -r tsconfig-paths/register ./__tests/admin", + "tests:makeit-example": "ts-node-dev --respawn --transpileOnly -r tsconfig-paths/register ./__tests/makeit-example" + }, + "author": "Sanawar Syed - Makeit", + "license": "MIT", + "devDependencies": { + "@types/graphql": "^14.5.0", + "@types/graphql-iso-date": "^3.4.0", + "@types/luxon": "^1.26.2", + "@types/mongodb": "^3.6.10", + "@types/node": "^14.14.35", + "@typescript-eslint/eslint-plugin": "^4.18.0", + "@typescript-eslint/parser": "^4.18.0", + "apollo-server": "^2.21.1", + "awesome-typescript-loader": "^5.2.1", + "cache-loader": "^4.1.0", + "copy-webpack-plugin": "^8.0.0", + "eslint-config-airbnb-base": "^14.2.1", + "eslint-plugin-import": "^2.22.1", + "execa": "^5.0.0", + "fork-ts-checker-webpack-plugin": "^6.1.1", + "graphql-zeus": "^3.0.5", + "listr": "^0.14.3", + "serverless": "^2.30.3", + "serverless-domain-manager": "^5.1.0", + "serverless-offline": "^6.8.0", + "serverless-webpack": "^5.4.0", + "ts-node": "^9.1.1", + "ts-node-dev": "^1.1.6", + "tsconfig-paths": "^3.9.0", + "typescript": "^4.2.3", + "webpack": "^5.26.3", + "webpack-node-externals": "^2.5.2" + }, + "dependencies": { + "@googlemaps/google-maps-services-js": "^3.1.16", + "apollo-server-lambda": "^2.21.1", + "axios": "^0.21.1", + "class-transformer": "^0.4.0", + "class-validator": "^0.13.1", + "dataloader": "^2.0.0", + "dotenv": "^8.2.0", + "file-type": "^16.3.0", + "firebase-admin": "^9.5.0", + "geolib": "^3.3.1", + "googleapis": "^67.1.1", + "graphql": "^15.5.0", + "graphql-iso-date": "^3.6.1", + "graphql-type-json": "^0.3.2", + "jimp": "^0.16.1", + "lodash": "^4.17.21", + "luxon": "^1.26.0", + "moment": "^2.29.1", + "moment-timezone": "^0.5.33", + "mongodb": "^3.6.5", + "node-fetch": "^2.6.1", + "node-geocoder": "^3.27.0", + "node-mailjet": "^3.3.1", + "reflect-metadata": "^0.1.13", + "retry-axios": "^2.4.0", + "serverless-http": "^2.7.0", + "stripe": "^8.138.0", + "twilio": "^3.58.0", + "type-graphql": "^1.1.1", + "uuid": "^8.3.2" + } +} diff --git a/serverless.yaml b/serverless.yaml new file mode 100644 index 0000000..1ce56d4 --- /dev/null +++ b/serverless.yaml @@ -0,0 +1,44 @@ +app: ${file(./package.json):name} +<<<<<<< HEAD +service: cronjob +======= +service: hooks +>>>>>>> develop +package: + individually: false +provider: + name: aws + stage: ${opt:stage,'dev'} + runtime: nodejs12.x + environment: + AWS_compute: ${self:service}-${self:provider.stage}-compute + iamRoleStatements: + - Effect: Allow + Action: + - lambda:InvokeFunction + Resource: '*' +plugins: + - serverless-webpack +custom: + stage: ${opt:stage, self:provider.stage} + webpack: + webpackConfig: './lib/seed/webpack.config.js' # Name of webpack configuration file + includeModules: + forceExclude: + - aws-sdk + - puppeteer + +functions: +<<<<<<< HEAD + removeDraftBookings: + handler: lib/__cronjobs/handler.removeDraftBookings + timeout: 300 # optional, in seconds, default is 6 + events: + - schedule: rate(1 minute) +======= + hookHandler: + handler: handler.hookHandler + memorySize: 1024 # optional, in MB, default is 1024 + timeout: 600 # optional, in seconds, default is 6 + async: true +>>>>>>> develop diff --git a/services/api-messaging/README.md b/services/api-messaging/README.md new file mode 100644 index 0000000..71b12f8 --- /dev/null +++ b/services/api-messaging/README.md @@ -0,0 +1,6 @@ +# api-messaging + +# Make-it messaging service +[![](https://ci6.googleusercontent.com/proxy/2HHmNs0zn0uGJb9RqML73pX6Bmd-BntMaIYR6IpXeIHKqn2_tG60C1pLNtMHoLHOk1I5CT8k5x9gvb-C1zuudHnD5HDOpvHljVYcKmDL--rb0Ar8IT4UB7YaG1c=s0-d-e1-ft#https://makeit-assets.s3.eu-central-1.amazonaws.com/signature_sanawar.png)](https://makeit-studio.com) + +![Build Status](https://travis-ci.org/joemccann/dillinger.svg?branch=master) diff --git a/services/api-messaging/__tests/playground.gql b/services/api-messaging/__tests/playground.gql new file mode 100644 index 0000000..325f355 --- /dev/null +++ b/services/api-messaging/__tests/playground.gql @@ -0,0 +1,67 @@ +mutation addRoom { + roomsCreateOne(input: { accountIds: ["kbRiSJEjHbX3KOEcF5IK41M4SgH2"] }) { + _id + title + accountIds + getAccounts { + _id + userName + } + } +} + +query myRooms { + roomsGetMany { + _id + title + accountIds + getAccounts { + _id + userName + } + } +} + +query myRoom { + roomsGetOne(id: "8e8099a2-5645-448e-981e-8a97c5beee5f") { + _id + title + accountIds + getAccounts { + _id + userName + } + } +} + +mutation addMessage { + roomsAddOneMessage(input: { roomId: "8e8099a2-5645-448e-981e-8a97c5beee5f", message: "Hello again" }) { + _id + message + sentBy + } +} + +query getOneRoomMessages { + roomsGetOneMessages(roomId: "8e8099a2-5645-448e-981e-8a97c5beee5f", pagination: { sort: "createdAt desc" }) { + _id + message + readBy { + accountId + readAt + } + createdAt + } +} + +mutation markMessageAsRead { + roomsMarkMessageAsRead(id: "ea279bcf-52c7-4ccd-b04f-1a586ebd15e5") { + _id + message + sentBy + readBy { + accountId + readAt + } + } +} diff --git a/services/api-messaging/app.ts b/services/api-messaging/app.ts new file mode 100644 index 0000000..1b52890 --- /dev/null +++ b/services/api-messaging/app.ts @@ -0,0 +1,74 @@ +import 'reflect-metadata'; +import { buildTypeDefsAndResolvers, buildSchema, NonEmptyArray } from 'type-graphql'; + +import env from '@config/.env.json'; + +import { authMiddleware, errorMiddleware, ctxMiddleware, complexityMiddleware } from '@seed/graphql/Middleware'; + +import Firebase from '@seed/services/auth/FirebaseService'; +import { ApiMessagingResolver } from './handler'; +import { SettingsCache } from '@seed/graphql/Settings'; + +// eslint-disable-next-line @typescript-eslint/no-var-requires +const { ApolloServer } = require('apollo-server'); + +const PORT = process.env.PORT || 4000; + +const bootstrap = async (): Promise => { + const settingsI = SettingsCache.getInstance(); + await settingsI.refreshCache(); + try { + // // eslint-disable-next-line @typescript-eslint/no-unused-vars + // let { typeDefs, resolvers } = await buildTypeDefsAndResolvers({ + // resolvers: AppResolvers, + // authChecker: authMiddleware, + // }); + + // typeDefs = typeDefs.replace('scalar Upload', 'scalar UploadFix'); + // resolvers = resolvers; + + const schema = await buildSchema({ + resolvers: (ApiMessagingResolver as unknown) as NonEmptyArray, + authChecker: authMiddleware, + }); + + const server = new ApolloServer({ + // typeDefs, + // resolvers, + schema, + formatError: errorMiddleware, + formatResponse: (response): any => { + return response; + }, + context: ctxMiddleware, + tracing: true, + plugins: [ + { + // eslint-disable-next-line @typescript-eslint/explicit-function-return-type + requestDidStart: () => ({ + // eslint-disable-next-line @typescript-eslint/explicit-function-return-type + didResolveOperation({ request, document }) { + return complexityMiddleware(schema, request.variables, document); + }, + }), + }, + ], + }); + + server.listen(PORT).then(({ url }) => { + console.log(`🚀 Server ready at ${url}`); + }); + + try { + const token = await Firebase.getInstance().createTokenId(process.env.TEST_USER_EMAIL ? process.env.TEST_USER_EMAIL : ''); + console.log(`Token for ${process.env.TEST_USER_EMAIL}`, token.idToken); + } catch (error) { + console.log('no token available'); + } + } catch (error) { + console.log('[GRAPH ERROR]', error); + throw error; + } +}; + +bootstrap(); diff --git a/services/api-messaging/components.ts b/services/api-messaging/components.ts new file mode 100644 index 0000000..9867ced --- /dev/null +++ b/services/api-messaging/components.ts @@ -0,0 +1,35 @@ +/* + ███████╗███╗ ██╗██╗ ██╗███╗ ███╗ + ██╔════╝████╗ ██║██║ ██║████╗ ████║ + █████╗ ██╔██╗ ██║██║ ██║██╔████╔██║ + ██╔══╝ ██║╚██╗██║██║ ██║██║╚██╔╝██║ + ███████╗██║ ╚████║╚██████╔╝██║ ╚═╝ ██║ + ╚══════╝╚═╝ ╚═══╝ ╚═════╝ ╚═╝ ╚═╝ +*/ + +import { ObjectType, Field } from 'type-graphql'; + +/* + + ██████╗ ██████╗ ███╗ ███╗██████╗ ██████╗ ███╗ ██╗███████╗███╗ ██╗████████╗███████╗ +██╔════╝██╔═══██╗████╗ ████║██╔══██╗██╔═══██╗████╗ ██║██╔════╝████╗ ██║╚══██╔══╝██╔════╝ +██║ ██║ ██║██╔████╔██║██████╔╝██║ ██║██╔██╗ ██║█████╗ ██╔██╗ ██║ ██║ ███████╗ +██║ ██║ ██║██║╚██╔╝██║██╔═══╝ ██║ ██║██║╚██╗██║██╔══╝ ██║╚██╗██║ ██║ ╚════██║ +╚██████╗╚██████╔╝██║ ╚═╝ ██║██║ ╚██████╔╝██║ ╚████║███████╗██║ ╚████║ ██║ ███████║ + ╚═════╝ ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═════╝ ╚═╝ ╚═══╝╚══════╝╚═╝ ╚═══╝ ╚═╝ ╚══════╝ + +*/ + +@ObjectType() +export class UnreadCount { + @Field() + accountId: string; + @Field() + count: number; +} + +@ObjectType() +export class RoomsStats { + @Field(() => [UnreadCount]) + unreadCounts: UnreadCount[]; +} diff --git a/services/api-messaging/functions/messages/message.model.ts b/services/api-messaging/functions/messages/message.model.ts new file mode 100644 index 0000000..f18c075 --- /dev/null +++ b/services/api-messaging/functions/messages/message.model.ts @@ -0,0 +1,91 @@ +import { ObjectType, Field, ID, ArgsType, InputType, Int } from 'type-graphql'; + +import { BaseGraphModel } from '@seed/graphql/BaseGraphModel'; +import ImageComponent from '@seed/interfaces/components'; +import { GetArgs } from '@seed/graphql/Request'; + +import { Permission } from '@seed/interfaces/permission'; +import { AccountTypeEnum } from '@src/accounts/account.components'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.admin], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export class MessageReadBy { + @Field() + accountId: string; + + @Field() + readAt: Date; +} + +@ObjectType() +export default class MessageModel extends BaseGraphModel { + public constructor() { + super({ + // ...init, + collectionName: 'messaging.messages', + permissions: permissions, + }); + } + // eslint-disable-next-line @typescript-eslint/no-unused-vars + @Field(() => ID) + readonly _id: string; + + @Field() + roomId: string; + + @Field({nullable: true}) + notified?: boolean; + + @Field() + message: string; + @Field() + edited: boolean; + @Field() + deleted: boolean; + + @Field(() => ImageComponent, { nullable: true }) + file: ImageComponent; + + @Field() + sentBy: string; + + @Field(() => [MessageReadBy]) + readBy: MessageReadBy[]; + + searchOptions(): string[] { + return []; + } + filterOptions(): string[] { + return []; + } +} + +@ArgsType() +export class MessageArgs { + @Field(() => GetArgs, { nullable: true }) + pagination?: GetArgs; +} + +@InputType() +export class NewMessageInput implements Partial { + @Field() + roomId: string; + + @Field() + message: string; + + @Field(() => ImageComponent, { nullable: true }) + file?: ImageComponent; +} + +@InputType() +export class EditMessageInput implements Partial { + @Field() + message: string; +} diff --git a/services/api-messaging/functions/messages/message.resolver.ts b/services/api-messaging/functions/messages/message.resolver.ts new file mode 100644 index 0000000..01fce63 --- /dev/null +++ b/services/api-messaging/functions/messages/message.resolver.ts @@ -0,0 +1,161 @@ +import { Resolver, Query, FieldResolver, Root, Arg, Mutation, Ctx, Args, Authorized } from 'type-graphql'; + +import { getOneGeneric, getManyGenericWithArgs, addOneGeneric, editOneGeneric, deleteOneGeneric } from '@seed/graphql/BaseService'; +import MessageModel, { MessageArgs, NewMessageInput, EditMessageInput } from '@services/api-messaging/functions/messages/message.model'; + +import RoomResolver from '@services/api-messaging/functions/rooms/room.resolver'; +import { ApolloContext } from '@seed/interfaces/context'; +import { ApolloError } from 'apollo-server-lambda'; +import _ from 'lodash'; + +@Resolver(MessageModel) +export default class MessageResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ + + @FieldResolver() + message(@Root() mess: MessageModel, @Ctx() ctx: ApolloContext): string { + if (mess.deleted) return '[DELETED]'; + return mess.message; + } + + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + @Mutation(() => MessageModel) + @Authorized() + async roomsAddOneMessage(@Arg('input') input: NewMessageInput, @Ctx() ctx: ApolloContext): Promise { + try { + // Get the Room first + const roomResolver = new RoomResolver(); + const room = await roomResolver.roomsGetOne(input.roomId, ctx); + + const model = new MessageModel(); + const result = model.saveOne( + { + ...input, + edited: false, + sentBy: ctx.ctx.user._id, + readBy: [], + r: room.r, + w: [ctx.ctx.user._id], + d: [ctx.ctx.user._id], + createdAt: new Date(), + updatedAt: new Date(), + }, + ctx, + ); + + // Todo - Refactor + await room.updateUnreadCounts(model, 1); + + await room.updateOne( + { _id: room._id }, + { + lastMessage: new Date(), + }, + null, + ); + + return result; + } catch (error) { + throw error; + } + } + + @Mutation(() => MessageModel) + @Authorized() + async roomsEditOneMessage(@Arg('id') id: string, @Arg('input') input: EditMessageInput, @Ctx() ctx: ApolloContext): Promise { + const model = new MessageModel(); + const message = await model.getOne({ _id: id }, ctx); + + if (message.sentBy !== ctx.ctx.user._id) + throw new ApolloError('permissions.notAllowed', '400', { you: ctx.ctx.user._id, allowed: message.sentBy }); + + return model.updateOne( + { _id: id }, + { + message: input.message, + edited: true, + updatedAt: new Date(), + }, + ctx, + ); + } + + @Mutation(() => MessageModel) + @Authorized() + async roomsMarkMessageAsRead(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = new MessageModel(); + const message = await model.getOne({ _id: id }, ctx); + + // The hypothesis here is that the message are accessible only by concerned people. + // No need to double check with the rooms accountsId + + const newReadBy = _.uniqBy(message.readBy.concat({ accountId: ctx.ctx.user._id, readAt: new Date() }), 'accountId'); + const result = await model.updateOne( + { _id: id }, + { + readBy: newReadBy, + }, + null, + ); + + // Update stats & notify + // Get the Room first + const roomResolver = new RoomResolver(); + const room = await roomResolver.roomsGetOne(result.roomId, ctx); + + await room.updateUnreadCounts(result, -1); + + return result; + } + + @Mutation(() => MessageModel) + @Authorized() + async roomsDeleteOneMessage(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = new MessageModel(); + const message = await model.getOne({ _id: id }, ctx); + + if (message.sentBy !== ctx.ctx.user._id) + throw new ApolloError('permissions.notAllowed', '400', { you: ctx.ctx.user._id, allowed: message.sentBy }); + + const result = await model.updateOne( + { _id: id }, + { + deleted: true, + updatedAt: new Date(), + }, + ctx, + ); + + // Update stats & notify + // Get the Room first + const roomResolver = new RoomResolver(); + const room = await roomResolver.roomsGetOne(result.roomId, ctx); + + await room.updateUnreadCounts(result, -1); + + return result; + } +} diff --git a/services/api-messaging/functions/rooms/room.model.ts b/services/api-messaging/functions/rooms/room.model.ts new file mode 100644 index 0000000..b492dd1 --- /dev/null +++ b/services/api-messaging/functions/rooms/room.model.ts @@ -0,0 +1,115 @@ +import { ObjectType, Field, ID, ArgsType, InputType, Int } from 'type-graphql'; + +import { BaseGraphModel } from '@seed/graphql/BaseGraphModel'; +import { Permission } from '@seed/interfaces/permission'; +import AccountModel from '@src/accounts/account.model'; +import { GetArgs } from '@seed/graphql/Request'; +import { ApolloContext } from '@seed/interfaces/context'; +import { ApolloError } from 'apollo-server-lambda'; +import { Stats } from 'fs'; +import { RoomsStats } from '@services/api-messaging/components'; +import MessageModel from '../messages/message.model'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import _ from 'lodash'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.admin, AccountTypeEnum.public], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class RoomModel extends BaseGraphModel { + public constructor() { + super({ + // ...init, + collectionName: 'messaging.rooms', + permissions: permissions, + }); + } + // eslint-disable-next-line @typescript-eslint/no-unused-vars + @Field(() => ID) + readonly _id: string; + + @Field(() => String, { nullable: true }) + title?: string; + + @Field(() => RoomsStats) + stats: RoomsStats; + + @Field(() => [AccountModel], { nullable: true }) + getAccounts?: AccountModel[]; + + @Field(() => [MessageModel], { nullable: true }) + getMessages?: MessageModel[]; + + @Field({ nullable: true }) + lastMessage: Date; + + searchOptions(): string[] { + return ['title']; + } + filterOptions(): string[] { + return ['title']; + } + + async updateUnreadCounts(newMessage: MessageModel, count: 1 | -1 | 0): Promise { + const { sentBy } = newMessage; + for (let index = 0; index < this.stats.unreadCounts.length; index++) { + // Add a count to all the other one + if (this.stats.unreadCounts[index].accountId != sentBy) { + if (count == 0) this.stats.unreadCounts[index].count = 0; + else this.stats.unreadCounts[index].count = this.stats.unreadCounts[index].count + count; + } + if (this.stats.unreadCounts[index].count < 0) this.stats.unreadCounts[index].count = 0; + } + + await (await this.db()).findOneAndUpdate( + { _id: this.get()._id }, + { $set: { stats: this.stats } as any }, + { upsert: false, returnOriginal: false }, + ); + } +} + +@ArgsType() +export class RoomArgs { + @Field(() => GetArgs, { nullable: true }) + pagination?: GetArgs; +} + +@InputType() +export class NewRoomInput { + @Field(() => [String]) + accountIds: string[]; +} + +@InputType() +export class AddPeopleToRoomInput { + @Field() + roomId: string; + @Field(() => [String]) + accountIds: string[]; +} + +@InputType() +export class RemovePeopleFromRoomInput { + @Field() + roomId: string; + @Field(() => [String]) + accountIds: string[]; +} + +export const onCreate = async (input: NewRoomInput, ctx: ApolloContext): Promise => { + try { + // To refactor + const users = await ctx.ctx.loaders.accountLoader.loadMany(input.accountIds); + + input.accountIds.forEach((element) => { + if (_.findIndex(users, { _id: element }) == -1) throw new ApolloError('users.notFound', '400', { _id: element }); + }); + } catch (error) { + throw error; + } +}; diff --git a/services/api-messaging/functions/rooms/room.resolver.ts b/services/api-messaging/functions/rooms/room.resolver.ts new file mode 100644 index 0000000..f25e6ed --- /dev/null +++ b/services/api-messaging/functions/rooms/room.resolver.ts @@ -0,0 +1,211 @@ +import { Resolver, Query, FieldResolver, Root, Arg, Mutation, Ctx, Args, Authorized } from 'type-graphql'; + +import { addOneGeneric } from '@seed/graphql/BaseService'; +import RoomModel, { + RoomArgs, + RemovePeopleFromRoomInput, + AddPeopleToRoomInput, + NewRoomInput, +} from '@services/api-messaging/functions/rooms/room.model'; +import { ApolloContext } from '@seed/interfaces/context'; +import { ApolloError } from 'apollo-server-lambda'; +import AccountModel from '@src/accounts/account.model'; +import { onCreate } from '@services/api-messaging/functions/rooms/room.model'; +import _ from 'lodash'; +import MessageModel, { MessageArgs } from '../messages/message.model'; +import { oneToManyComplexity } from '@seed/graphql/Middleware'; +import { UnreadCount } from '@services/api-messaging/components'; + +@Resolver(RoomModel) +export default class RoomResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ + + @FieldResolver() + async getAccounts(@Root() room: RoomModel, @Ctx() ctx: ApolloContext): Promise<(AccountModel | Error)[]> { + return await ctx.ctx.loaders.accountLoader.loadMany(room.r); + } + @FieldResolver(() => [MessageModel], { complexity: oneToManyComplexity }) + async getMessages(@Args() args: MessageArgs, @Root() room: RoomModel, @Ctx() ctx: ApolloContext): Promise { + try { + if (!args.pagination) args.pagination = { limit: 10, skip: 0, sort: 'createdAt desc' }; + else if (args.pagination && !args.pagination.sort) args.pagination = { ...args.pagination, sort: 'createdAt desc' }; + + return new MessageModel().getMany({ roomId: room._id }, args.pagination, ctx); + } catch (error) { + throw error; + } + } + + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + @Query(() => RoomModel) + @Authorized() + async roomsGetOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = new RoomModel(); + const roomData = await model.getOne({ _id: id }, ctx); + if (!roomData.r.includes(ctx.ctx.user._id)) + throw new ApolloError('permissions.notAllowed', '400', { you: ctx.ctx.user._id, allowed: roomData.r }); + + return roomData; + } + + @Query(() => [RoomModel]) + @Authorized() + async roomsGetMany(@Args() args: RoomArgs, @Ctx() ctx: ApolloContext): Promise { + const model = new RoomModel(); + const result = await model.getMany({ accountIds: ctx.ctx.user._id }, args.pagination, ctx); + return result; + } + + @Query(() => [MessageModel]) + @Authorized() + async roomsGetMessages(@Arg('id') id: string, @Args() args: RoomArgs, @Ctx() ctx: ApolloContext): Promise { + const model = new RoomModel(); + const roomData = await model.getOne({ _id: id }, ctx); + if (!roomData.r.includes(ctx.ctx.user._id)) + throw new ApolloError('permissions.notAllowed', '400', { you: ctx.ctx.user._id, allowed: roomData.r }); + + if (!args.pagination) args.pagination = { limit: 10, skip: 0, sort: 'createdAt desc' }; + else if (args.pagination && !args.pagination.sort) args.pagination = { ...args.pagination, sort: 'createdAt desc' }; + + return new MessageModel().getMany({ roomId: id }, args.pagination, ctx); + } + + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + @Mutation(() => RoomModel) + @Authorized() + async roomsCreateOne(@Arg('input') input: NewRoomInput, @Ctx() ctx: ApolloContext): Promise { + const model = new RoomModel(); + + try { + await onCreate(input, ctx); + } catch (error) { + throw error; + } + + input.accountIds.push(ctx.ctx.user._id); + input.accountIds = _.uniq(input.accountIds); + + // Check if a room with the same user doesn't exists already + + try { + return await model.getOne( + { + r: { $all: input.accountIds }, + }, + ctx, + ); + } catch (error) { + const unreadCounts: UnreadCount[] = []; + for (let index = 0; index < input.accountIds.length; index++) { + unreadCounts.push({ + accountId: input.accountIds[index], + count: 0, + }); + } + + const dataToSave: Partial = { + ...input, + r: input.accountIds, + w: input.accountIds, + stats: { + unreadCounts: unreadCounts, + }, + }; + return addOneGeneric(model, dataToSave, ctx); + } + } + + @Mutation(() => RoomModel) + @Authorized() + async roomsAddPeople(@Arg('input') input: AddPeopleToRoomInput, @Ctx() ctx: ApolloContext): Promise { + const model = new RoomModel(); + + // Get the room first to see if allowed + const roomData = await model.getOne({ _id: input.roomId }, ctx); + if (!roomData.r.includes(ctx.ctx.user._id)) + throw new ApolloError('permissions.notAllowed', '400', { you: ctx.ctx.user._id, allowed: roomData.r }); + + const newAccountIds = _.uniq(roomData.r.concat(input.accountIds)); + + try { + await onCreate(input, ctx); + } catch (error) { + throw error; + } + + // TODO : Remove from stats + return roomData.updateOne({ _id: input.roomId }, { r: newAccountIds, w: newAccountIds }, ctx); + } + + @Mutation(() => RoomModel) + @Authorized() + async roomsRemovePeople(@Arg('input') input: RemovePeopleFromRoomInput, @Ctx() ctx: ApolloContext): Promise { + const model = new RoomModel(); + + // Get the room first to see if allowed + const roomData = await model.getOne({ _id: input.roomId }, ctx); + if (!roomData.r.includes(ctx.ctx.user._id)) + throw new ApolloError('permissions.notAllowed', '400', { you: ctx.ctx.user._id, allowed: roomData.r }); + + const newAccountIds = _.uniq( + roomData.r.filter(function(element) { + return roomData.r.indexOf(element) === -1; + }), + ); + + // TODO : Remove from stats + + return roomData.updateOne({ _id: input.roomId }, { r: newAccountIds, w: newAccountIds }, ctx); + } + + @Mutation(() => RoomModel) + @Authorized() + async roomsMarkAllMessageAsRead(@Arg('roomId') roomId: string, @Ctx() ctx: ApolloContext): Promise { + const roomModel = new RoomModel(); + await roomModel.getOne({ _id: roomId }, ctx); + + // The hypothesis here is that the message are accessible only by concerned people. + const model = new MessageModel(); + await (await model.db()).updateMany( + { roomId: roomId, 'readBy.accountId': { $ne: ctx.ctx.user._id } }, + { + $push: { readBy: { accountId: ctx.ctx.user._id, readAt: new Date() } }, + }, + ); + + // Update stats & notify + for (let index = 0; index < roomModel.stats.unreadCounts.length; index++) { + // Add a count to all the other one + if (roomModel.stats.unreadCounts[index].accountId == ctx.ctx.user._id) roomModel.stats.unreadCounts[index].count = 0; + } + + await (await roomModel.db()).findOneAndUpdate( + { _id: roomModel._id }, + { $set: { stats: roomModel.stats } }, + { upsert: false, returnOriginal: false }, + ); + + return roomModel; + } +} diff --git a/services/api-messaging/functions/rooms/room.service.ts b/services/api-messaging/functions/rooms/room.service.ts new file mode 100644 index 0000000..5d18430 --- /dev/null +++ b/services/api-messaging/functions/rooms/room.service.ts @@ -0,0 +1,17 @@ +import { ApolloContext } from '@seed/interfaces/context'; +import RoomModel from './room.model'; +import RoomResolver from './room.resolver'; + +export const roomsGetOneByAccountsIds = async (ids: string[], toCreate?: { ctx: ApolloContext }): Promise => { + const model = new RoomModel(); + + try { + const roomData = await model.getOne({ r: { $all: ids } }, null); + + return roomData; + } catch (error) { + if (!toCreate) throw error; + + return await new RoomResolver().roomsCreateOne({ accountIds: ids }, toCreate.ctx); + } +}; diff --git a/services/api-messaging/serverless-messaging.yaml b/services/api-messaging/serverless-messaging.yaml new file mode 100644 index 0000000..f057034 --- /dev/null +++ b/services/api-messaging/serverless-messaging.yaml @@ -0,0 +1,40 @@ +app: ${file(./package.json):name} +service: messaging +package: + individually: true +provider: + name: aws + stage: ${opt:stage,'dev'} + runtime: nodejs10.x + environment: + AWS_compute: ${self:service}-${self:provider.stage}-compute + iamRoleStatements: + - Effect: Allow + Action: + - lambda:InvokeFunction + Resource: '*' +plugins: + - serverless-webpack + - serverless-domain-manager +custom: + stage: ${opt:stage, self:provider.stage} + domains: ${file(./config/domains.yaml):domains} + customDomain: ${file(./config/domains.yaml):customDomain} + webpack: + webpackConfig: './lib/seed/webpack.config.js' # Name of webpack configuration file + includeModules: + forceExclude: + - aws-sdk +functions: + handler: + handler: handler.handler + timeout: 30 + events: + - http: + path: / + method: post + cors: true + - http: + path: / + method: get + cors: true diff --git a/services/api-uploads/README.md b/services/api-uploads/README.md new file mode 100644 index 0000000..8e23cc8 --- /dev/null +++ b/services/api-uploads/README.md @@ -0,0 +1,9 @@ +# Make-it Upload Service + +[![](https://ci6.googleusercontent.com/proxy/2HHmNs0zn0uGJb9RqML73pX6Bmd-BntMaIYR6IpXeIHKqn2_tG60C1pLNtMHoLHOk1I5CT8k5x9gvb-C1zuudHnD5HDOpvHljVYcKmDL--rb0Ar8IT4UB7YaG1c=s0-d-e1-ft#https://makeit-assets.s3.eu-central-1.amazonaws.com/signature_sanawar.png)](https://makeit-studio.com) + +![Build Status](https://travis-ci.org/joemccann/dillinger.svg?branch=master) + +### Documentation + +https://docs.google.com/document/d/1Nw-9-QFOXBXZxrxHPCv6vojtwedWN2hVsihQr-NIza4/edit# diff --git a/services/api-uploads/__tests/playground.gql b/services/api-uploads/__tests/playground.gql new file mode 100644 index 0000000..c31eaef --- /dev/null +++ b/services/api-uploads/__tests/playground.gql @@ -0,0 +1,25 @@ +# Write your query or mutation here +mutation uploadFile { + uploadFile( + input: { + title: "myFile" + base64: "" + } + ) { + _id + large + medium + small + title + } +} + +query myFiles { + files { + _id + large + medium + small + title + } +} diff --git a/services/api-uploads/app.ts b/services/api-uploads/app.ts new file mode 100644 index 0000000..47e27d9 --- /dev/null +++ b/services/api-uploads/app.ts @@ -0,0 +1,79 @@ +import 'reflect-metadata'; +import { buildTypeDefsAndResolvers, buildSchema, NonEmptyArray } from 'type-graphql'; + +import { authMiddleware, errorMiddleware, ctxMiddleware, complexityMiddleware } from '@seed/graphql/Middleware'; + +import Firebase from '@seed/services/auth/FirebaseService'; +import { ApiUploadResolver } from './handler'; +import { SettingsCache } from '@seed/graphql/Settings'; + +// eslint-disable-next-line @typescript-eslint/no-var-requires +const { ApolloServer } = require('apollo-server'); + +const PORT = process.env.PORT || 4000; + +const bootstrap = async (): Promise => { + const settingsI = SettingsCache.getInstance(); + await settingsI.refreshCache(); + + try { + // // eslint-disable-next-line @typescript-eslint/no-unused-vars + // let { typeDefs, resolvers } = await buildTypeDefsAndResolvers({ + // resolvers: AppResolvers, + // authChecker: authMiddleware, + // }); + + // typeDefs = typeDefs.replace('scalar Upload', 'scalar UploadFix'); + // resolvers = resolvers; + + const schema = await buildSchema({ + resolvers: ([...ApiUploadResolver] as unknown) as NonEmptyArray, + authChecker: authMiddleware, + }); + + const server = new ApolloServer({ + // typeDefs, + // resolvers, + schema, + formatError: errorMiddleware, + formatResponse: (response): any => { + return response; + }, + context: ctxMiddleware, + tracing: true, + plugins: [ + { + // eslint-disable-next-line @typescript-eslint/explicit-function-return-type + requestDidStart: () => ({ + // eslint-disable-next-line @typescript-eslint/explicit-function-return-type + didResolveOperation({ request, document }) { + return complexityMiddleware(schema, request.variables, document); + }, + }), + }, + ], + }); + + const token = await Firebase.getInstance().createTokenId(process.env.TEST_ADMIN_EMAIL ? process.env.TEST_ADMIN_EMAIL : ''); + console.log(token); + + server.listen(PORT).then(({ url }) => { + console.log(`🚀 Server ready at ${url}`); + }); + + // // await listCustomFields(); + // const addres = await addContact({ + // email: 'sanawar@makeit-studio.com', + // firstName: 'Sanawar', + // lastName: 'Syed', + // phone: '+32484740444', + // }); + + // await editContactFields(addres, { stage: 'Phase 3' }); + } catch (error) { + console.log('[GRAPH ERROR]', error); + throw error; + } +}; + +bootstrap(); diff --git a/services/api-uploads/file.model.ts b/services/api-uploads/file.model.ts new file mode 100644 index 0000000..971f026 --- /dev/null +++ b/services/api-uploads/file.model.ts @@ -0,0 +1,92 @@ +import { ObjectType, Field, ID, ArgsType, InputType, registerEnumType } from 'type-graphql'; + +import { BaseGraphModel } from '@seed/graphql/BaseGraphModel'; +import { Permission } from '@seed/interfaces/permission'; + +import { GetArgs } from '@seed/graphql/Request'; + +import AccountModel from '@src/accounts/account.model'; +import { AccountTypeEnum } from '@src/accounts/account.components'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.public], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +export enum FileType { + image = 'image', + document = 'document', +} +registerEnumType(FileType, { + name: 'FileType', +}); + +@ObjectType() +export default class FileModel extends BaseGraphModel { + public constructor() { + super({ + // ...init, + collectionName: 'files', + permissions: permissions, + }); + } + // eslint-disable-next-line @typescript-eslint/no-unused-vars + @Field(() => ID) + readonly _id: string; + + @Field() + title: string; + + @Field() + accountId: string; + + @Field(() => AccountModel, { nullable: true }) + account?: AccountModel; + + @Field() + fileType: FileType; + + @Field() + fileName: string; + + @Field() + large: string; + + @Field({ nullable: true }) + medium?: string; + + @Field({ nullable: true }) + small?: string; + + @Field({ nullable: true }) + fileNameMedium?: string; + + @Field({ nullable: true }) + fileNameSmall?: string; + + searchOptions(): string[] { + return ['title']; + } + filterOptions(): string[] { + return ['title']; + } +} + +@ArgsType() +export class FileArgs { + @Field(() => GetArgs, { nullable: true }) + pagination?: GetArgs; +} +@InputType() +export class NewFileInput { + @Field() + title: string; + + @Field() + base64: string; + + @Field({ nullable: true }) + privateFile?: boolean; +} diff --git a/services/api-uploads/file.resolver.ts b/services/api-uploads/file.resolver.ts new file mode 100644 index 0000000..d51a6b3 --- /dev/null +++ b/services/api-uploads/file.resolver.ts @@ -0,0 +1,59 @@ +import { Resolver, Query, Args, Authorized, Ctx } from 'type-graphql'; +import { Mutation, Arg } from 'type-graphql'; + +import { getOneGeneric, getManyGenericWithArgs } from '@seed/graphql/BaseService'; +import { ApolloContext } from '@seed/interfaces/context'; + +import FileModel, { FileArgs, NewFileInput } from '@services/api-uploads/file.model'; +import { uploadFileWithBase64 } from '@services/api-uploads/services/UploadService'; + +@Resolver(FileModel) +export default class FileResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + @Query(() => FileModel) + @Authorized('admin') + async file(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = new FileModel(); + return getOneGeneric(model, id, ctx); + } + + @Query(() => [FileModel]) + @Authorized('admin') + async files(@Args() fileArgs: FileArgs, @Ctx() ctx: ApolloContext): Promise { + const model = new FileModel(); + return getManyGenericWithArgs(model, fileArgs, ctx); + } + + @Query(() => [FileModel]) + @Authorized() + async myFiles(@Args() fileArgs: FileArgs, @Ctx() ctx: ApolloContext): Promise { + const model = new FileModel(); + const args = { ...fileArgs, accountId: ctx.ctx.user._id }; + return getManyGenericWithArgs(model, args, ctx); + } + + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + @Mutation(() => FileModel) + @Authorized() + async uploadFile(@Arg('input') input: NewFileInput, @Ctx() ctx: ApolloContext): Promise { + try { + return await uploadFileWithBase64(input.base64, input.title, input.privateFile || false, ctx); + } catch (error) { + throw error; + } + } +} diff --git a/services/api-uploads/serverless-upload.yaml b/services/api-uploads/serverless-upload.yaml new file mode 100644 index 0000000..855575f --- /dev/null +++ b/services/api-uploads/serverless-upload.yaml @@ -0,0 +1,40 @@ +app: ${file(./package.json):name} +service: uploads +package: + individually: true +provider: + name: aws + stage: ${opt:stage,'dev'} + runtime: nodejs12.x + environment: + S3_BUCKET: ${self:app}-uploads-production + iamRoleStatements: + - Effect: Allow + Action: + - s3:* + Resource: arn:aws:s3:::${self:provider.environment.S3_BUCKET}/* +plugins: + - serverless-webpack + - serverless-domain-manager +custom: + stage: ${opt:stage, self:provider.stage} + domains: ${file(./config/domains.yaml):domains} + customDomain: ${file(./config/domains.yaml):customDomain} + webpack: + webpackConfig: './lib/seed/webpack.config.js' # Name of webpack configuration file + includeModules: + forceExclude: + - aws-sdk +functions: + fileHandler: + handler: handler.filesHandler + timeout: 30 # optional, in seconds, default is 6timeout: 10 # optional, in seconds, default is 6 + events: + - http: + path: / + method: post + cors: true + - http: + path: / + method: get + cors: true diff --git a/services/api-uploads/services/ImageService.ts b/services/api-uploads/services/ImageService.ts new file mode 100644 index 0000000..dffc5b3 --- /dev/null +++ b/services/api-uploads/services/ImageService.ts @@ -0,0 +1,53 @@ +import Jimp from 'jimp'; + +export const resize = async ( + input: Buffer, + mimeType: string, +): Promise<{ + // extension: string; + // mimeType: string; + largeBuffer: Buffer; + mediumBuffer: Buffer; + smallBuffer: Buffer; +}> => { + const resolutions = { + large: parseInt(process.env.IMG_largeRes || '2000'), + medium: parseInt(process.env.IMG_mediumRes || '1200'), + small: parseInt(process.env.IMG_smallRes || '600'), + }; + + const quality = parseInt(process.env.IMG_quality || '80'); + + try { + const jimpLarge = await Jimp.read(input); + const jimpMedium = await jimpLarge.clone(); + const jimpSmall = await jimpLarge.clone(); + + if (jimpLarge.getWidth() > resolutions.large) { + jimpLarge.resize(resolutions.large, Jimp.AUTO); + } + const largeBuffer = jimpLarge.quality(quality).getBufferAsync(mimeType); + + if (jimpMedium.getWidth() > resolutions.medium) { + jimpMedium.resize(resolutions.medium, Jimp.AUTO); + } + const mediumBuffer = jimpMedium.quality(quality).getBufferAsync(mimeType); + + if (jimpSmall.getWidth() > resolutions.small) { + jimpSmall.resize(resolutions.small, Jimp.AUTO); + } + const smallBuffer = jimpSmall.quality(quality).getBufferAsync(mimeType); + + const result = { + // extension: extension, + // mimeType: mimeType, + largeBuffer: await largeBuffer, + mediumBuffer: await mediumBuffer, + smallBuffer: await smallBuffer, + }; + + return result; + } catch (error) { + throw error; + } +}; diff --git a/services/api-uploads/services/S3Service.ts b/services/api-uploads/services/S3Service.ts new file mode 100644 index 0000000..5a44330 --- /dev/null +++ b/services/api-uploads/services/S3Service.ts @@ -0,0 +1,57 @@ +import AWS from 'aws-sdk'; +const s3 = new AWS.S3(); + +// eslint-disable-next-line @typescript-eslint/class-name-casing +export interface UploadParams { + fileName: string; + contentType: string; + buffer: Buffer | string; + bucket?: string; + acl?: boolean; +} + +export interface SignedUrlParams { + fileName: string; + bucket?: string; +} + +export const uploadToS3 = async (param: UploadParams): Promise => { + const bucket = param.bucket || process.env.S3_BUCKET || 'app-uploads'; + const region = process.env.awsRegion || 'eu-east-1'; + + AWS.config.update({ region: region }); + + const s3Params: AWS.S3.PutObjectRequest = { + Body: param.buffer, + Bucket: bucket, + Key: param.fileName, + ContentType: param.contentType, + ACL: param.acl ? 'private' : 'public-read', + }; + try { + const uploadResult = await s3.upload(s3Params).promise(); + + return uploadResult.Location; //`https://${bucket}.s3-${region}.amazonaws.com/${fileName}`; + } catch (error) { + throw error; + } +}; + +export const getSignedS3Url = async (param: SignedUrlParams, signedUrlExpireSeconds?: number): Promise => { + const bucket = param.bucket || process.env.S3_BUCKET || 'app-uploads'; + const region = process.env.awsRegion || 'eu-east-1'; + + AWS.config.update({ region: region }); + + try { + const url = s3.getSignedUrl('getObject', { + Bucket: bucket, + Key: param.fileName, + Expires: signedUrlExpireSeconds || 60 * 5, + }); + + return url; + } catch (error) { + throw error; + } +}; diff --git a/services/api-uploads/services/UploadService.ts b/services/api-uploads/services/UploadService.ts new file mode 100644 index 0000000..c0f440a --- /dev/null +++ b/services/api-uploads/services/UploadService.ts @@ -0,0 +1,133 @@ +import { v4 as uuid } from 'uuid'; + +import FileTypeHelper from 'file-type'; +import core from 'file-type/core'; + +import { uploadToS3 } from '@services/api-uploads/services/S3Service'; +import { resize } from '@services/api-uploads/services/ImageService'; + +import FileModel, { FileType } from '@services/api-uploads/file.model'; +import { ApolloContext } from '@seed/interfaces/context'; + +export const uploadFileWithBuffer = async ( + buffer: Buffer, + contentType: { mime: string; ext: string }, + title: string, + privateFile: boolean, + ctx: ApolloContext, +): Promise => { + try { + const _id = uuid(); + + const fileName = title + '-' + _id + '.' + contentType?.ext; + + const large = uploadToS3({ + acl: privateFile, + buffer: buffer, + contentType: contentType?.mime || '', + fileName, + }); + + const file = await new FileModel().saveOne( + { + _id, + title, + fileType: FileType.image, + accountId: ctx.ctx.user._id, + large: await large, + fileName, + }, + ctx, + ); + return file; + } catch (error) { + throw error; + } +}; + +export const uploadImageWithBuffer = async ( + buffer: Buffer, + contentType: { mime: string; ext: string }, + title: string, + privateFile: boolean, + ctx: ApolloContext, +): Promise => { + try { + const allImages = await resize(buffer, contentType.mime); + const _id = uuid(); + const fileName = title + '-' + _id + '.' + contentType?.ext; + const fileNameMedium = title + '-' + _id + '-medium.' + contentType?.ext; + const fileNameSmall = title + '-' + _id + '-small.' + contentType?.ext; + + const large = uploadToS3({ + acl: privateFile, + buffer: allImages.largeBuffer, + contentType: contentType?.mime, + fileName, + }); + const medium = uploadToS3({ + acl: privateFile, + buffer: allImages.mediumBuffer, + contentType: contentType?.mime, + fileName: fileNameMedium, + }); + const small = uploadToS3({ + acl: privateFile, + buffer: allImages.smallBuffer, + contentType: contentType?.mime, + fileName: fileNameSmall, + }); + + const file = await new FileModel().saveOne( + { + _id, + title, + fileType: FileType.image, + accountId: ctx.ctx.user._id, + large: await large, + medium: await medium, + small: await small, + fileName, + fileNameMedium, + fileNameSmall, + }, + ctx, + ); + return file; + } catch (error) { + throw error; + } +}; + +export const uploadFileWithBase64 = async (base64: string, title: string, privateFile: boolean, ctx: ApolloContext): Promise => { + try { + const cleanBase64 = base64.split('base64,').pop() || ''; + const buffer = Buffer.from(cleanBase64, 'base64'); + const contentTypeCheck = await FileTypeHelper.fromBuffer(buffer); + + const contentType = { + mime: '', + ext: '', + }; + + // Check extensions to deal with SVG + const extension = title.split('.').pop(); + if (extension == 'svg') { + contentType.mime = 'image/svg+xml'; + contentType.ext = 'svg'; + } else { + if (!contentTypeCheck?.mime) throw new Error('cannotReadExtensions'); + contentType.mime = contentTypeCheck.mime; + contentType.ext = contentTypeCheck.ext; + } + + const allowedMime = process.env.FILE_allowedMime || 'image/jpeg,image/png,image/gif,application/pdf'; + if (!allowedMime.split(',').includes(contentType.mime)) throw new Error('badExtensions'); + + if (contentType.mime.includes('image') && contentType.mime != 'image/svg+xml') + return await uploadImageWithBuffer(buffer, contentType, title, privateFile, ctx); + else return await uploadFileWithBuffer(buffer, contentType, title, privateFile, ctx); + } catch (error) { + throw error; + } +}; diff --git a/services/module-accounts/.gitignore b/services/module-accounts/.gitignore new file mode 100644 index 0000000..9bea433 --- /dev/null +++ b/services/module-accounts/.gitignore @@ -0,0 +1,2 @@ + +.DS_Store diff --git a/services/module-accounts/README.md b/services/module-accounts/README.md new file mode 100644 index 0000000..a887982 --- /dev/null +++ b/services/module-accounts/README.md @@ -0,0 +1,19 @@ +# Make-it Account Module + +[![](https://ci6.googleusercontent.com/proxy/2HHmNs0zn0uGJb9RqML73pX6Bmd-BntMaIYR6IpXeIHKqn2_tG60C1pLNtMHoLHOk1I5CT8k5x9gvb-C1zuudHnD5HDOpvHljVYcKmDL--rb0Ar8IT4UB7YaG1c=s0-d-e1-ft#https://makeit-assets.s3.eu-central-1.amazonaws.com/signature_sanawar.png)](https://makeit-studio.com) + +![Build Status](https://travis-ci.org/joemccann/dillinger.svg?branch=master) + +### Description + +Adds the basic user management function based on + +- @src/accounts/account.model +- @src/accounts/account.components; + +### Install + +extend the resolvers + +export default class AccountAdminResolver extends AccountAdminBaseResolver +export default class AccountResolver extends AccountBaseResolver diff --git a/services/module-accounts/__tests/accounts.test.ts b/services/module-accounts/__tests/accounts.test.ts new file mode 100644 index 0000000..9d2db10 --- /dev/null +++ b/services/module-accounts/__tests/accounts.test.ts @@ -0,0 +1,74 @@ +/* eslint-disable @typescript-eslint/explicit-function-return-type */ +import { clog } from '../../../lib/seed/helpers/Utils'; +import { thunder } from '../../../__tests/app/config'; + +export class AccountEngineTest { + /* LOGIN */ + logInMagicLink = async (input: { email?: string }) => { + const { email } = input; + + const logInMagicLink = await thunder.query({ + logInMagicLink: [ + { + input: { email: email || 'software@makeit-studio.com' }, + }, + { + message: true, + }, + ], + }); + return logInMagicLink; + }; + + logInEmailCodeRequest = async (input: { email?: string }) => { + const { email } = input; + const logInEmailCodeRequest = await thunder.query({ + logInEmailCodeRequest: [ + { + input: { email: email || 'software@makeit-studio.com' }, + }, + { + message: true, + }, + ], + }); + + return logInEmailCodeRequest; + }; + + logInEmailCodeConfirm = async (input: { email?: string; code?: number }) => { + const { email, code } = input; + + const logInEmailCodeConfirm = await thunder.mutation({ + logInEmailCodeConfirm: [ + { + input: { email: email || 'software@makeit-studio.com', code: code || 0 }, + }, + { + idToken: true, + }, + ], + }); + + return logInEmailCodeConfirm; + }; + + /* SIGNIN */ + + signInEmailCodeRequest = async (input: { email?: string }) => { + const { email } = input; + + const signInEmailCodeRequest = await thunder.mutation({ + signInEmailCodeRequest: [ + { + input: { email: email || 'software@makeit-studio.com' }, + }, + { + message: true, + }, + ], + } as any); + + return signInEmailCodeRequest; + }; +} diff --git a/services/module-accounts/__tests/admin.playground.graphql b/services/module-accounts/__tests/admin.playground.graphql new file mode 100644 index 0000000..ccf1354 --- /dev/null +++ b/services/module-accounts/__tests/admin.playground.graphql @@ -0,0 +1,5 @@ +mutation resetPassword { + resetPassword(input: { email: "sanawar@makeit-studio.com" }) { + message + } +} diff --git a/services/module-accounts/__tests/playground.graphql b/services/module-accounts/__tests/playground.graphql new file mode 100644 index 0000000..bdd1420 --- /dev/null +++ b/services/module-accounts/__tests/playground.graphql @@ -0,0 +1,45 @@ +query login { + login(creds: { email: "", password: "" }) { + idToken + refreshToken + } +} + +query loginToken { + login(refreshToken: "") { + idToken + refreshToken + } +} + +mutation resetPassword { + resetPassword(input: { email: "sanawar@makeit-studio.com" }) { + message + } +} + +mutation updateMe { + updateMe(input: { settings: [{ type: notifications, email: true }, { type: promotions, sms: false, pushNotifications: true }] }) { + _id + firstName + lastName + settings { + type + email + pushNotifications + sms + } + } +} + +mutation signInEmailCodeRequest { + signInEmailCodeRequest(input: { email: "sanawar@makeit-studio.com" }) { + message + } +} + +mutation signInEmailCodeConfirmm { + signInEmailCodeConfirmm(input: { email: "sanawar@makeit-studio.com", code: 140773 }) { + idToken + } +} diff --git a/services/module-accounts/account.components.ts b/services/module-accounts/account.components.ts new file mode 100644 index 0000000..8136f4f --- /dev/null +++ b/services/module-accounts/account.components.ts @@ -0,0 +1,140 @@ +import { InputType, Field, ObjectType } from 'type-graphql'; +import { SettingsTypeEnum } from '@src/accounts/account.components'; +import { AvailableTranslation } from '@src/__components/components'; + +@InputType() +export class LoginInput { + @Field() + email: string; + @Field() + password: string; +} + +@InputType() +export class LinkEmailInput extends LoginInput { + @Field() + idToken: string; +} + +@InputType() +export class ResetPasswordInput { + @Field() + email: string; +} + +@InputType() +export class NewPasswordInput { + @Field() + oldPassword: string; + + @Field() + newPassword: string; + + @Field() + newPasswordConfirmation: string; +} + +@InputType() +export class NewEmailInput { + @Field() + newEmail: string; +} + +@InputType() +export class SMSVerifInput { + @Field() + email: string; + @Field() + emailCode: string; + @Field() + recaptchaToken: string; +} + +export class SMSVerif { + @Field() + phoneNumber: string; + @Field() + recaptchaToken: string; +} + +@InputType('SettingsInput') +@ObjectType() +export class SettingsComponent { + @Field(() => SettingsTypeEnum) + type: SettingsTypeEnum; + + @Field({ nullable: true }) + email?: boolean; + + @Field({ nullable: true }) + pushNotifications?: boolean; + + @Field({ nullable: true }) + sms?: boolean; +} + +@InputType() +export class CodeRequestInput { + @Field({ nullable: true }) + firstName?: string; + @Field({ nullable: true }) + lastName?: string; + @Field() + email: string; + + @Field({ nullable: true }) + phoneNumber?: string; + + @Field(() => AvailableTranslation, { nullable: true }) + language?: AvailableTranslation; +} + +@InputType() +export class CodeConfirmInput { + @Field() + email: string; + @Field() + code: number; +} + +@ObjectType() +export class CodeRequest { + @Field() + code: number; + @Field() + expiresAt: Date; +} + +// eslint-disable-next-line @typescript-eslint/class-name-casing +export class googleAuthToken { + refresh_token: string; + access_token: string; + scope: string; + token_type: string; + expiry_date: number; +} + +export class OAuthTokens { + googleTokens: googleAuthToken; +} + +export class SecurityStatusComponent { + emailVerified: boolean; + phoneVerified: boolean; +} + +export enum SecurityLevelEnum { + 'nothing' = 'nothing', // Nothing + 'email-only' = 'email-only', // Only Email + 'email-phone' = 'email-phone', // Email and phone + 'email-mfa' = 'email-mfa', // Email and other authenticator +} +// @ObjectType() +// export class ProfileCompletion { +// @Field() +// type: AccountTypeEnum; +// @Field() +// completed: boolean; +// @Field() +// currentStep: string; +// } diff --git a/services/module-accounts/account.model.ts b/services/module-accounts/account.model.ts new file mode 100644 index 0000000..4e18116 --- /dev/null +++ b/services/module-accounts/account.model.ts @@ -0,0 +1,258 @@ +import { ObjectType, Field, ID, ArgsType, Int, InputType } from 'type-graphql'; + +import { BaseGraphModel } from '@seed/graphql/BaseGraphModel'; +import { Permission } from '@seed/interfaces/permission'; + +import ImageComponent, { AccountGenderEnum } from '@seed/interfaces/components'; +import { AddressComponent } from '@seed/interfaces/components.geo'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { AvailableTranslation } from '@src/__components/components'; +import { GetArgs } from '@seed/graphql/Request'; +import { CodeRequest, OAuthTokens, SecurityLevelEnum, SecurityStatusComponent, SettingsComponent } from './account.components'; + +@ObjectType() +export default class BaseAccountModel extends BaseGraphModel { + public constructor(permissions: Permission) { + super({ + // ...init, + collectionName: 'accounts', + permissions: permissions, + }); + } + + // eslint-disable-next-line @typescript-eslint/no-unused-vars + @Field(() => ID) + readonly _id: string; + + @Field(() => [AccountTypeEnum]) + types: AccountTypeEnum[]; + + @Field(() => [String], { nullable: true }) + organisationIds?: string[]; + + @Field() + email: string; + + @Field({ nullable: true }) + userName?: string; + + @Field({ nullable: true }) + firstName?: string; + + @Field({ nullable: true }) + lastName?: string; + + @Field(() => ImageComponent, { nullable: true }) + profilePicture?: ImageComponent; + + @Field({ nullable: true }) + phoneNumber?: string; + + @Field(() => AccountGenderEnum, { nullable: true }) + gender?: AccountGenderEnum; + + // @Field(() => Int, { nullable: true }) + // age?: number; + + @Field({ nullable: true }) + birthDate?: Date; + + @Field(() => Int, { nullable: true }) + getAge(): number | undefined { + if (this.birthDate) { + const today = new Date(); + let age = today.getFullYear() - this.birthDate.getFullYear(); + const diff = today.getMonth() - this.birthDate.getMonth(); + if (diff < 0 || (diff === 0 && today.getDate() < this.birthDate.getDate())) { + age--; + } + return age; + } + } + + @Field(() => AddressComponent, { nullable: true }) + address?: AddressComponent; + + @Field(() => AvailableTranslation, { nullable: true }) + language: AvailableTranslation; + + @Field(() => [SettingsComponent], { nullable: true }) + settings?: SettingsComponent[]; + + /* + For Authorization codes + */ + authCodes?: CodeRequest[]; + /* + For OauthProvider codes + */ + oAuthTokens?: OAuthTokens; + + security?: SecurityStatusComponent; + @Field() + securityCheck(): boolean { + // Check the level of security for this app + const securityLevel: SecurityLevelEnum = (process.env.SECURITY_LEVEL as SecurityLevelEnum) || SecurityLevelEnum.nothing; + + switch (securityLevel) { + case SecurityLevelEnum.nothing: + console.error('[SECURITY] NO CONFIG'); + break; + case SecurityLevelEnum['email-only']: + if (this.security?.emailVerified) return true; + else return false; + case SecurityLevelEnum['email-phone']: + if (this.security?.emailVerified && this.security?.phoneVerified) return true; + else return false; + case SecurityLevelEnum['email-mfa']: + console.error('[SECURITY] NOT IMPLEMENTED'); + break; + default: + console.error('[SECURITY] BAD CONFIGURATION'); + break; + } + + return true; + } + + // [x: string]: any; + + searchOptions(): string[] { + return ['email', 'firstName', 'lastName', 'userName']; + } + + filterOptions(): string[] { + return ['email', 'firstName', 'lastName', 'userName']; + } +} + +@ArgsType() +export class BaseAccountArgs { + @Field(() => GetArgs, { nullable: true }) + pagination?: GetArgs; + + @Field(() => AccountTypeEnum) + types: AccountTypeEnum; +} + +@InputType() +export class NewAccountInputAdmin implements Partial { + @Field(() => AccountTypeEnum) + type: AccountTypeEnum; + + @Field(() => [String], { nullable: true }) + organisationIds?: string[]; + + @Field() + email: string; + + // @Field() + // password: string; + + // @Field() + // passwordConfirmation: string; + + @Field({ nullable: true }) + userName?: string; + + @Field({ nullable: true }) + firstName?: string; + + @Field({ nullable: true }) + lastName?: string; + + @Field(() => ImageComponent, { nullable: true }) + profilePicture?: ImageComponent; + + @Field({ nullable: true }) + phoneNumber?: string; + + @Field(() => AccountGenderEnum, { nullable: true }) + gender?: AccountGenderEnum; + + @Field({ nullable: true }) + birthDate?: Date; + + @Field(() => AddressComponent, { nullable: true }) + address?: AddressComponent; + @Field(() => AvailableTranslation, { nullable: true }) + language?: AvailableTranslation; + + @Field(() => [SettingsComponent], { nullable: true }) + settings?: SettingsComponent[]; +} + +@InputType() +export class NewBaseAccountInput implements Partial { + @Field() + email: string; + + @Field() + password: string; + + @Field() + passwordConfirmation: string; + + @Field({ nullable: true }) + userName?: string; + + @Field({ nullable: true }) + firstName?: string; + + @Field({ nullable: true }) + lastName?: string; + + @Field(() => ImageComponent, { nullable: true }) + profilePicture?: ImageComponent; + + @Field({ nullable: true }) + phoneNumber?: string; + + @Field(() => AccountGenderEnum, { nullable: true }) + gender?: AccountGenderEnum; + + @Field({ nullable: true }) + birthDate?: Date; + + @Field(() => AddressComponent, { nullable: true }) + address?: AddressComponent; + + @Field(() => AvailableTranslation, { nullable: true }) + language?: AvailableTranslation; + + @Field(() => [SettingsComponent], { nullable: true }) + settings?: SettingsComponent[]; +} + +@InputType() +export class EditBaseAccountInput implements Partial { + @Field({ nullable: true }) + userName?: string; + + @Field({ nullable: true }) + firstName?: string; + + @Field({ nullable: true }) + lastName?: string; + + @Field(() => ImageComponent, { nullable: true }) + profilePicture?: ImageComponent; + + @Field({ nullable: true }) + phoneNumber?: string; + + @Field(() => AccountGenderEnum, { nullable: true }) + gender?: AccountGenderEnum; + + @Field({ nullable: true }) + birthDate?: Date; + + @Field(() => AddressComponent, { nullable: true }) + address?: AddressComponent; + + @Field(() => AvailableTranslation, { nullable: true }) + language?: AvailableTranslation; + + @Field(() => [SettingsComponent], { nullable: true }) + settings?: SettingsComponent[]; +} diff --git a/services/module-accounts/account.resolver.admin.ts b/services/module-accounts/account.resolver.admin.ts new file mode 100644 index 0000000..2ae9a9d --- /dev/null +++ b/services/module-accounts/account.resolver.admin.ts @@ -0,0 +1,125 @@ +import { Resolver, Query, Args, Ctx, Authorized } from 'type-graphql'; +import { Mutation, Arg } from 'type-graphql'; + +import { ApolloContext } from '@seed/interfaces/context'; +import { getOneGeneric, editOneGeneric, getManyGenericWithArgs } from '@seed/graphql/BaseService'; + +import { deleteOneGeneric, signinGenericNoPassword } from './services/AccountService'; + +import AccountModel, { EditAccountInput, AccountArgs } from '@src/accounts/account.model'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { NewAccountInputAdmin } from './account.model'; +import { AvailableTranslation } from '@src/__components/components'; +import { FirebaseTokenResult, SimpleResult } from '@seed/interfaces/components'; +import { LoginInput, ResetPasswordInput } from './account.components'; +import Firebase from '@seed/services/auth/FirebaseService'; +import AccountBaseResolver from './account.resolver'; +import { sendEmail } from '@seed/services/email/EmailService'; +import FirebaseAuth from '@seed/services/auth/FirebaseAuthService'; +import { ApolloError } from 'apollo-server-lambda'; +import { newError } from '@seed/helpers/Error'; + +@Resolver(AccountModel) +export default class AccountAdminBaseResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ + + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + @Query(() => AccountModel) + @Authorized() + async me(@Ctx() ctx: ApolloContext): Promise { + return ctx.ctx.user as AccountModel; + } + + /* + * Auth side + */ + @Query(() => FirebaseTokenResult) + async login( + @Arg('creds', { nullable: true }) creds?: LoginInput, + @Arg('refreshToken', { nullable: true }) refreshToken?: string, + ): Promise { + if (creds) return FirebaseAuth.getInstance().loginWithEmailPassword(creds); + if (refreshToken) return FirebaseAuth.getInstance().loginWithRefreshToken(refreshToken); + + throw newError(3000); + } + + @Query(() => AccountModel) + @Authorized(AccountTypeEnum.admin) + async accountsGetOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = new AccountModel(); + return getOneGeneric(model, id, ctx); + } + + @Query(() => [AccountModel]) + @Authorized(AccountTypeEnum.admin) + async accountsGetMany(@Args() accountArgs: AccountArgs, @Ctx() ctx: ApolloContext): Promise { + const model = new AccountModel(); + return getManyGenericWithArgs(model, accountArgs, ctx); + } + + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + + /* + * Admin side + */ + + @Mutation(() => AccountModel) + @Authorized(AccountTypeEnum.admin) + async accountsAddOne(@Arg('input') input: NewAccountInputAdmin, @Ctx() ctx: ApolloContext): Promise { + const model = new AccountModel(); + const result = await signinGenericNoPassword(model, input.type, input, ctx); + const resetPasswordLink = await Firebase.getInstance().getResetPasswordLink(result.email); + + await sendEmail('adminAccountAddOne', result.language || AvailableTranslation.en, result.email, { ...result, resetPasswordLink }); + + return result as any; + } + + @Mutation(() => AccountModel) + @Authorized(AccountTypeEnum.admin) + async accountsEditOne(@Arg('id') id: string, @Arg('input') input: EditAccountInput, @Ctx() ctx: ApolloContext): Promise { + const model = new AccountModel(); + return editOneGeneric(model, id, input, ctx); + } + + @Mutation(() => AccountModel) + @Authorized(AccountTypeEnum.admin) + async accountsDeleteOne( + @Arg('id') id: string, + @Arg('userType', () => AccountTypeEnum) userType: AccountTypeEnum, + @Ctx() ctx: ApolloContext, + ): Promise { + const model = new AccountModel(); + return deleteOneGeneric(model, userType, id, ctx); + } + + /* + * ACTIONS + */ + @Mutation(() => SimpleResult) + async resetPassword(@Arg('input') input: ResetPasswordInput, @Ctx() ctx: ApolloContext): Promise { + return new AccountBaseResolver().resetPassword(input, ctx); + } +} diff --git a/services/module-accounts/account.resolver.ts b/services/module-accounts/account.resolver.ts new file mode 100644 index 0000000..fcadb10 --- /dev/null +++ b/services/module-accounts/account.resolver.ts @@ -0,0 +1,171 @@ +import { ApolloError } from 'apollo-server-lambda'; +import { Resolver, Query, Ctx, Authorized, Mutation, Arg } from 'type-graphql'; + +import { ApolloContext } from '@seed/interfaces/context'; +import { editOneGeneric } from '@seed/graphql/BaseService'; + +import { signinAnonymous, signinGeneric } from './services/AccountService'; + +import AccountModel, { EditAccountInput } from '@src/accounts/account.model'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { FirebaseTokenResult, SimpleResult } from '@seed/interfaces/components'; +import { ResetPasswordInput, NewEmailInput, NewPasswordInput, LoginInput, LinkEmailInput } from './account.components'; +import Firebase from '@seed/services/auth/FirebaseService'; +import { sendEmail } from '@seed/services/email/EmailService'; +import { AvailableTranslation } from '@src/__components/components'; +import FirebaseAuth from '@seed/services/auth/FirebaseAuthService'; +import { newError } from '@seed/helpers/Error'; + +@Resolver(AccountModel) +export default class AccountBaseResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ + + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + @Query(() => AccountModel) + @Authorized() + async me(@Ctx() ctx: ApolloContext): Promise { + return ctx.ctx.user as AccountModel; + } + + /* + * Auth side + */ + @Query(() => FirebaseTokenResult) + async login( + @Ctx() ctx: ApolloContext, + @Arg('creds', { nullable: true }) creds?: LoginInput, + @Arg('refreshToken', { nullable: true }) refreshToken?: string, + ): Promise { + if (creds) { + const account = new AccountModel(); + await account.getOne({ email: creds.email }, null); + + return FirebaseAuth.getInstance().loginWithEmailPassword(creds); + } + if (refreshToken) return FirebaseAuth.getInstance().loginWithRefreshToken(refreshToken); + + return signinAnonymous(ctx); + } + + @Query(() => SimpleResult) + async magicLink(@Arg('input') input: ResetPasswordInput, @Ctx() ctx: ApolloContext): Promise { + const account = new AccountModel(); + + try { + await account.getOne({ email: input.email }, null); + const magicLink = await Firebase.getInstance().getMagicLink(input.email); + + await sendEmail('magicLink', account.language, account.email, { ...account, magicLink }); + } catch (error) { + throw error; + } + + return { message: 'sent' }; + } + + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + + /* + * Account side + */ + + // @Mutation(() => AccountModel) + // async register(@Arg('input') input: NewAccountInput, @Ctx() ctx: ApolloContext): Promise { + // const model = new AccountModel(); + // return signinGeneric(model, AccountTypeEnum.user, input, ctx); + // } + + @Mutation(() => AccountModel) + @Authorized() + async updateMe(@Arg('input') input: EditAccountInput, @Ctx() ctx: ApolloContext): Promise { + const model = new AccountModel(); + return editOneGeneric(model, ctx.ctx.user._id, input, ctx); + } + + @Mutation(() => AccountModel) + @Authorized() + async updateMeEmail(@Arg('input') input: NewEmailInput, @Ctx() ctx: ApolloContext): Promise { + const currentUser = ctx.ctx.user as AccountModel; + try { + await Firebase.getInstance().updateEmail({ uid: currentUser._id, email: input.newEmail }); + return editOneGeneric(currentUser, currentUser._id, { email: input.newEmail }, ctx); + } catch (error) { + throw error; + } + } + + @Mutation(() => AccountModel) + @Authorized() + async updateMePassword(@Arg('input') input: NewPasswordInput, @Ctx() ctx: ApolloContext): Promise { + const currentUser = ctx.ctx.user as AccountModel; + + if (input.newPassword != input.newPasswordConfirmation) throw newError(2006); + + try { + // Check old password + FirebaseAuth.getInstance().loginWithEmailPassword({ + email: currentUser.email, + password: input.oldPassword, + }); + } catch (error) { + throw newError(20060); + } + + try { + await Firebase.getInstance().updatePassword({ uid: currentUser._id, password: input.newPassword }); + return currentUser; + } catch (error) { + throw error; + } + } + + @Mutation(() => SimpleResult) + async resetPassword(@Arg('input') input: ResetPasswordInput, @Ctx() ctx: ApolloContext): Promise { + const account = new AccountModel(); + + try { + await account.getOne({ email: input.email }, null); + const resetPasswordLink = await Firebase.getInstance().getResetPasswordLink(input.email); + + await sendEmail('resetPassword', account.language, account.email, { ...account, resetPasswordLink }); + } catch (error) { + throw error; + } + + return { message: 'sent' }; + } + + @Mutation(() => AccountModel) + @Authorized('public') + async registerGuest( + @Arg('input') input: LinkEmailInput, + @Arg('otherInfo') otherInfo: EditAccountInput, + @Ctx() ctx: ApolloContext, + ): Promise { + const account = new AccountModel(); + + const result = await FirebaseAuth.getInstance().linkWithEmailPassword(input); + return editOneGeneric(account, result.localId, { ...otherInfo, email: result.email, types: [AccountTypeEnum.user] }, ctx); + } +} diff --git a/services/module-accounts/env/emails.settings.import.json b/services/module-accounts/env/emails.settings.import.json new file mode 100644 index 0000000..1d000a7 --- /dev/null +++ b/services/module-accounts/env/emails.settings.import.json @@ -0,0 +1,229 @@ + + +/* +* EMAILS +*/ + +{ + "_id" : "04e48fa2-714f-4d7f-ae17-c791ccf6e5bb", + "key" : "resetPassword", + "templateId" : { + "en" : "", + "fr" : "", + "nl" : "" + }, + "fromEmail" : "no-reply@makeit-studio.com", + "replyToEmail" : "software@makeit-studio.com", + "fromName" : "Make it | Studio", + "subject" : { + "en" : "You requested a new password", + "fr" : "Vous avez demandé un nouveau mot de passe", + "nl" : "You requested a new password" + }, + "body" : { + "en" : "

Here is the password reset link : ${resetPasswordLink}

", + "fr" : "

Voici le lien pour réinitialiser votre mot de passe : ${resetPasswordLink}

", + "nl" : "

Here is the password reset link : ${resetPasswordLink}

" + }, + "availableFields" : [ + "firstName", + "lastName", + "email", + "resetPasswordLink" + ], + "custom" : false, + "type" : "emails", + "cci" : [ + + ], + "r" : [ + "admin" + ], + "w" : [ + "admin" + ], + "d" : [ + "admin" + ], + "createdAt" : ISODate(), + "updatedAt" : ISODate(), +}, +{ + "_id" : "0f245c6c-e39a-465f-9374-e2268a0cbf83", + "key" : "adminAccountAddOne", + "templateId" : { + "en" : "", + "fr" : "", + "nl" : "" + }, + "fromEmail" : "no-reply@makeit-studio.com", + "replyToEmail" : "software@makeit-studio.com", + "fromName" : "Make it | Studio", + "subject" : { + "en" : "The administrator of [PROJECT] created an account for you", + "fr" : "L'administrateur de [PROJECT] a crée un compte pour vous!", + "nl" : "The administrator of [PROJECT] created an account for you" + }, + "body" : { + "en" : "

Please setup your password by clicking this link : ${resetPasswordLink}

", + "fr" : "

Pouvez-vous configurer un mot de passe en cliquant sur ce lien ? : ${resetPasswordLink}

", + "nl" : "

Please setup your password by clicking this link : ${resetPasswordLink}

" + }, + "availableFields" : [ + "firstName", + "lastName", + "email", + "resetPasswordLink" + ], + "custom" : false, + "type" : "emails", + "cci" : [ + + ], + "r" : [ + "admin" + ], + "w" : [ + "admin" + ], + "d" : [ + "admin" + ], + "createdAt" : ISODate(), + "updatedAt" : ISODate(), +}, +{ + "_id" : "16496dfc-5935-495b-a352-52efe158d55c", + "key" : "organisationsAddOneAccount", + "templateId" : { + "en" : "", + "fr" : "", + "nl" : "" + }, + "fromEmail" : "no-reply@makeit-studio.com", + "replyToEmail" : "software@makeit-studio.com", + "fromName" : "Make it | Studio", + "subject" : { + "en" : "The administrator of [ORG] created an account for you", + "fr" : "L'administrateur de l'organisation [ORG] a crée un compte pour vous!", + "nl" : "The administrator of [ORG] created an account for you" + }, + "body" : { + "en" : "

Please setup your password by clicking this link : ${resetPasswordLink}

", + "fr" : "

Pouvez-vous configurer un mot de passe en cliquant sur ce lien ? : ${resetPasswordLink}

", + "nl" : "

Please setup your password by clicking this link : ${resetPasswordLink}

" + }, + "availableFields" : [ + "firstName", + "lastName", + "email", + "resetPasswordLink" + ], + "custom" : false, + "type" : "emails", + "cci" : [ + + ], + "r" : [ + "admin" + ], + "w" : [ + "admin" + ], + "d" : [ + "admin" + ], + "createdAt" : ISODate(), + "updatedAt" : ISODate(), +}, +{ + "_id" : "16496dfc-5875-495b-axx2-52efe158d55c", + "key" : "emailCodeSignIn", + "templateId" : { + "en" : "", + "fr" : "", + "nl" : "" + }, + "fromEmail" : "no-reply@makeit-studio.com", + "replyToEmail" : "software@makeit-studio.com", + "fromName" : "Make it | Studio", + "subject" : { + "en" : "Here is your code to login : ${data.code}", + "fr" : "Voici votre code d'accès: ${data.code}", + "nl" : "Here is your code to login : ${data.code}" + }, + "body" : { + "en" : "Here is your code to login : ${data.code}", + "fr" : "Voici votre code d'accès: ${data.code}", + "nl" : "Here is your code to login : ${data.code}" + }, + "availableFields" : [ + "firstName", + "code" + ], + "custom" : false, + "type" : "emails", + "cci" : [ + + ], + "r" : [ + "admin" + ], + "w" : [ + "admin" + ], + "d" : [ + "admin" + ], + "createdAt" : ISODate(), + "updatedAt" : ISODate(), +}, +{ + "_id" : "04e48fa2-714f-4d7f-ae17-c791ccf6e5bb", + "key" : "magicLink", + "templateId" : { + "en" : "", + "fr" : "", + "nl" : "" + }, + "subject" : { + "en" : "Your magic link is ready", + "fr" : "Votre demande de lien d'accès", + "nl" : "Your magic link is ready" + }, + "body" : { + "en" : "Here is your link to login : ${magicLink}", + "fr" : "Voici votre lien d'accès: ${magicLink}", + "nl" : "Here is your link to login : ${magicLink}" + }, + "fromEmail" : "no-reply@makeit-studio.com", + "replyToEmail" : "software@makeit-studio.com", + "fromName" : "Make it | Studio", + "availableFields" : [ + "firstName", + "lastName", + "email", + "magicLink" + ], + "custom" : false, + "type" : "emails", + "cci" : [ + + ], + "r" : [ + "admin" + ], + "w" : [ + "admin" + ], + "d" : [ + "admin" + ], + "createdAt" : ISODate(), + "updatedAt" : ISODate(), +} + +MAGIC_LINK_URL + +IOS_PACKAGE_NAME +ANDROID_PACKAGE_NAME \ No newline at end of file diff --git a/services/module-accounts/env/env.settings.import.json b/services/module-accounts/env/env.settings.import.json new file mode 100644 index 0000000..6f69cc8 --- /dev/null +++ b/services/module-accounts/env/env.settings.import.json @@ -0,0 +1,51 @@ +{ + "_id" : "44ae9d0c-75d5-4c4f-8a74-020104b3d0d5", + "key" : "MAGIC_LINK_URL", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "d" : [ + "admin", + ], + "r" : [ + "admin", + ], + "w" : [ + "admin", + ] +}, +{ + "_id" : "987cd596-a364-4d84-a423-18f549595cf6", + "key" : "IOS_PACKAGE_NAME", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "d" : [ + "admin", + ], + "r" : [ + "admin", + ], + "w" : [ + "admin", + ] +}, +{ + "_id" : "b48d4e97-ddcf-4b7b-89fd-d7844071563f", + "key" : "ANDROID_PACKAGE_NAME", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin", + ], + "w" : [ + "admin", + ], + "d" : [ + "admin", + ], +}, \ No newline at end of file diff --git a/services/module-accounts/functions/auth-history/pages.model.ts b/services/module-accounts/functions/auth-history/pages.model.ts new file mode 100644 index 0000000..5203a81 --- /dev/null +++ b/services/module-accounts/functions/auth-history/pages.model.ts @@ -0,0 +1,51 @@ +// import { ObjectType, Field, ID, ArgsType, InputType, registerEnumType, Int } from 'type-graphql'; + +// import { BaseGraphModel } from '@seed/graphql/BaseGraphModel'; +// import { Permission } from '@seed/interfaces/permission'; + +// import { GetArgs } from '@seed/graphql/Request'; + +// import { GraphQLJSONObject } from 'graphql-type-json'; +// import { AccountTypeEnum } from '@src/accounts/account.components'; +// import { EngineModel } from '@seed/engine/EngineModel'; +// import { PageDBInterfaceSchema, PageDBSchema, PageSchema } from './schemas/auth-history.schema'; +// import { plainToClass } from 'class-transformer'; +// import { IEngineSchema } from '@seed/engine/EngineSchema'; +// import { uploadToS3 } from '@services/api-uploads/services/S3Service'; +// import { updateCaching } from '@services/module-cms/services/PageService'; +// import { StreamDBSchema } from '@seed/engine/utils/streams/schemas/stream.schema'; + +// const permissions: Permission = { +// c: [AccountTypeEnum.admin], +// r: [AccountTypeEnum.public], +// w: [AccountTypeEnum.admin], +// d: [AccountTypeEnum.admin], +// }; + +// @ObjectType() +// export default class PageModel extends EngineModel { +// public constructor(input?: PageDBInterfaceSchema & Partial) { +// const dataInit = plainToClass(PageDBSchema, input || {}); +// super({ +// collectionName: 'pages', +// permissions: permissions, +// dataInit, +// }); +// } + +// plainToClass(plain: any): PageSchema | PageSchema[] { +// return plainToClass(PageSchema, plain); +// } + +// async afterCreate(): Promise { +// await updateCaching(this); +// } + +// async afterUpdate(): Promise { +// await updateCaching(this); +// } + +// async afterDelete(): Promise { +// await updateCaching(this); +// } +// } diff --git a/services/module-accounts/functions/auth-history/pages.resolver.admin.ts b/services/module-accounts/functions/auth-history/pages.resolver.admin.ts new file mode 100644 index 0000000..9722872 --- /dev/null +++ b/services/module-accounts/functions/auth-history/pages.resolver.admin.ts @@ -0,0 +1,72 @@ +// import { Resolver, Query, Args, Authorized, Ctx } from 'type-graphql'; +// import { Mutation, Arg } from 'type-graphql'; + +// import { getOneGeneric, getManyGenericWithArgs, addOneGeneric, editOneGeneric, deleteOneGeneric } from '@seed/graphql/BaseService'; + +// import { AccountTypeEnum } from '@src/accounts/account.components'; +// import { createEngineMutationResolver, createEngineQueryResolver } from '@seed/engine/genericResolvers/BaseEngineResolver'; +// import { PageSchema } from './schemas/auth-history.schema'; +// import { PageArgs, NewPageInput, EditPageInput } from './schemas/pages.input'; +// import PageModel from './pages.model'; +// import { SuccessResponse } from '@seed/interfaces/response'; +// import { EngineMiddleware } from '@seed/graphql/MiddlewareV2'; +// import { updateCaching as updatePagesCaching } from '@services/module-cms/services/PageService'; +// import { ApolloContext } from '@seed/interfaces/context'; + +// const PageBaseQueryAdminResolver = createEngineQueryResolver({ +// domain: 'pages', +// schemaName: PageSchema, +// modelName: PageModel, +// argsType: PageArgs, +// engineMiddleware: { +// authorization: [AccountTypeEnum.admin], +// }, +// }); + +// const PageBaseMutationAdminResolver = createEngineMutationResolver({ +// domain: 'pages', +// schemaName: PageSchema, +// modelName: PageModel, +// newInput: NewPageInput, +// editInput: EditPageInput, +// engineMiddleware: { +// authorization: [AccountTypeEnum.admin], +// }, +// }); + +// @Resolver(PageModel) +// export class PageQueryAdminResolver extends PageBaseQueryAdminResolver { +// /* +// ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ +// ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ +// ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ +// ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ +// ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ +// ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ +// */ +// } + +// @Resolver(PageModel) +// export class PageMutationAdminResolver extends PageBaseMutationAdminResolver { +// /* +// ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ +// ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ +// ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ +// ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ +// ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ +// ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ +// */ + +// @Mutation(() => SuccessResponse) +// @EngineMiddleware({ +// authorization: [AccountTypeEnum.admin], +// }) +// async refreshCaching(@Ctx() ctx: ApolloContext): Promise { +// const model = new PageModel(); +// await updatePagesCaching(model); + +// return { success: true }; +// } +// } + +// export const PageAdminResolvers = [PageQueryAdminResolver, PageMutationAdminResolver]; diff --git a/services/module-accounts/functions/auth-history/schemas/auth-history.schema.ts b/services/module-accounts/functions/auth-history/schemas/auth-history.schema.ts new file mode 100644 index 0000000..9c8d03c --- /dev/null +++ b/services/module-accounts/functions/auth-history/schemas/auth-history.schema.ts @@ -0,0 +1,45 @@ +import { IEngineSchema, MetaBy, MetaPermissions } from '@seed/engine/EngineSchema'; +import { GraphQLJSONObject } from 'graphql-type-json'; +import { InputType, ObjectType, Field, Int } from 'type-graphql'; + +export enum AuthCodeType { + 'verification' = 'verification', + 'authentication' = 'authentication', +} + +export enum AuthCanalType { + 'email' = 'email', + 'phone' = 'phone', + 'authenticator' = 'authenticator', +} + +@ObjectType() +export class AuthHistoryBaseSchema { + @Field() + code: string; + + @Field(() => AuthCodeType) + type: AuthCodeType; + + @Field(() => AuthCanalType) + canal: AuthCanalType; +} + +@ObjectType() +export class AuthHistoryIDBSchema extends AuthHistoryBaseSchema { + @Field() + accountId: string; +} + +@ObjectType() +export class AuthHistoryDBSchema extends AuthHistoryIDBSchema {} + +@ObjectType({ implements: IEngineSchema }) +export class AuthHistorySchema extends AuthHistoryDBSchema implements IEngineSchema { + _id: string; + organisationId?: string | undefined; + by?: MetaBy | undefined; + permissions: MetaPermissions; + createdAt: Date; + updatedAt: Date; +} diff --git a/services/module-accounts/functions/auth-history/schemas/pages.input.ts b/services/module-accounts/functions/auth-history/schemas/pages.input.ts new file mode 100644 index 0000000..755f62c --- /dev/null +++ b/services/module-accounts/functions/auth-history/schemas/pages.input.ts @@ -0,0 +1,16 @@ +// import { InputType, ArgsType, Field } from 'type-graphql'; + +// import { GetManyArgs } from '@seed/graphql/Request'; +// import { PageBaseSchema } from './auth-history.schema'; + +// @InputType() +// export class NewPageInput extends PageBaseSchema {} + +// @InputType() +// export class EditPageInput extends PageBaseSchema { +// @Field({ nullable: true }) +// title: string; +// } + +// @ArgsType() +// export class PageArgs extends GetManyArgs {} diff --git a/services/module-accounts/resolvers/auth/account.auth.mfa.resolver.ts b/services/module-accounts/resolvers/auth/account.auth.mfa.resolver.ts new file mode 100644 index 0000000..4738915 --- /dev/null +++ b/services/module-accounts/resolvers/auth/account.auth.mfa.resolver.ts @@ -0,0 +1,246 @@ +/* eslint-disable @typescript-eslint/explicit-function-return-type */ +/* eslint-disable @typescript-eslint/no-use-before-define */ +import { Arg, Ctx, Mutation, Query, Resolver } from 'type-graphql'; +import _ from 'lodash'; +import { v4 as uuid } from 'uuid'; + +import AccountModel from '@src/accounts/account.model'; +import { CodeConfirmInput, CodeRequestInput, SMSVerif, SMSVerifInput } from '@services/module-accounts/account.components'; +import Firebase from '@lib/seed/services/auth/FirebaseService'; +import { sendEmail } from '@lib/seed/services/email/EmailService'; +import { FirebaseTokenResult, SimpleResult } from '@lib/seed/interfaces/components'; +import moment from 'moment'; +import { generateCodeNumber } from '@seed/helpers/Utils'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { ApolloContext } from '@seed/interfaces/context'; +import { newError } from '@seed/helpers/Error'; +import { AvailableTranslation } from '@src/__components/components'; +import { DateTime } from 'luxon'; +import FirebaseAuth from '@seed/services/auth/FirebaseAuthService'; +import { SuccessResponse } from '@seed/interfaces/response'; + +// TODO - Handle change emails & change phone number + +@Resolver(AccountModel) +export class AccountQueryMFAAuthResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + @Query(() => SimpleResult) + async sendEmailVerificationCode(@Arg('input') input: CodeRequestInput, @Ctx() ctx: ApolloContext): Promise { + try { + const magicLink = await Firebase.getInstance().getMagicLink(input.email); + await sendEmail('magicLinkLogIn', input.language || AvailableTranslation.en, input.email, { ...input, magicLink }); + return { message: 'sent' }; + } catch (error) { + throw error; + } + } + + @Query(() => SimpleResult) + async sendPhoneVerificationCode(@Arg('input') input: SMSVerifInput): Promise { + try { + // Check if account exists + + // Get Phone Number - TODO : google-libphonenumber + const phoneNumber = '+32484740444'; + // Get Code + + return await FirebaseAuth.getInstance().sendSMSVerification({ phoneNumber, recaptchaToken: input.recaptchaToken }); + } catch (error) { + throw error; + } + } + + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + + */ + + @Mutation(() => FirebaseTokenResult) + async logInEmailCodeConfirm(@Arg('input') input: CodeConfirmInput): Promise { + const { code, email } = input; + + try { + // Check if exists on firebase already + const { accountCheck, model, firebaseUserData } = await getAccount(email); + // Get the codes + const codes = accountCheck.authCodes; + + // verify the code + const currentCode = _.find(codes, { code: code }); + if (!currentCode) throw newError(2300); + + // verify the expiration date + if (DateTime.utc() >= DateTime.fromJSDate(currentCode.expiresAt).toUTC()) throw newError(2301); + + // Get the token from firebase + const token = await Firebase.getInstance().createTokenId(email); + + // remove it from the array + await model.updateOneCustom( + { _id: firebaseUserData.uid }, + { + $pull: { authCodes: { code: code } }, + }, + null, + ); + return token; + } catch (error) { + throw error; + } + } +} + +@Resolver(AccountModel) +export class AccountMutationCodeAuthResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + + */ +} + +async function getCode(model: AccountModel, data: AccountModel) { + // Get the codes + let codes = data.authCodes; + if (codes) { + // Remove the expired ones + _.remove(codes, (e) => { + const diffInMs = DateTime.fromJSDate(e.expiresAt).diffNow('milliseconds').milliseconds; + const MAX_TIMEOUT = 15 * 60 * 1000; + + if (diffInMs >= MAX_TIMEOUT) return true; + + return false; + }); + } else codes = []; + + // Check if too many attempt in the last 15 minutes + if (codes.length > 5) throw newError(2302); + + // Add the new one + const code = generateCodeNumber(); + + codes.push({ + code: code, + expiresAt: DateTime.utc() + .plus({ minutes: 15 }) + .toJSDate(), + }); + // Save in DB + await model.updateOneCustom( + { _id: data._id }, + { + $set: { authCodes: codes }, + }, + + null, + ); + return code; +} + +async function getAccount(email: string) { + let firebaseUserData; + try { + firebaseUserData = await Firebase.getInstance().getUserByEmail(email); + } catch (error) { + throw newError(2004, error); + } + + // Check if in DB or not + const model = new AccountModel(); + + const accountCheck = await (await model.db()).findOne({ _id: firebaseUserData.uid }); + // Verify if has thinkific signin + if (!accountCheck) throw newError(2004); + return { accountCheck, model, firebaseUserData }; +} + +async function createAccount(input: CodeRequestInput) { + let firebaseUserData; + try { + firebaseUserData = await Firebase.getInstance().getUserByEmail(input.email); + } catch (error) { + try { + firebaseUserData = await Firebase.getInstance().createUser({ + email: input.email, + password: uuid(), + }); + } catch (error) { + throw error; + } + } + + const dataToSave = { + ...input, + firebaseData: { + aud: firebaseUserData.aud, + emailVerified: firebaseUserData.email_verified, + firebase: JSON.stringify(firebaseUserData.firebase), + }, + updatedAt: new Date(), + }; + + // Check if in DB or not + const model = new AccountModel(); + const type = AccountTypeEnum.user; + + let account; + + const accountCheck = await (await model.db()).findOne({ _id: firebaseUserData.uid }); + // Verify if has thinkific signin + if (accountCheck) { + account = accountCheck; + await (await model.db()).updateOne( + { _id: firebaseUserData.uid }, + { + $set: { ...input }, + }, + ); + } else { + const types = [type]; + account = await model.saveOne( + { + ...dataToSave, + _id: firebaseUserData.uid, + types, + r: [AccountTypeEnum.admin, firebaseUserData.uid], + w: [AccountTypeEnum.admin, firebaseUserData.uid], + d: [AccountTypeEnum.admin, firebaseUserData.uid], + }, + null, + ); + } + return { model, firebaseUserData, account }; +} diff --git a/services/module-accounts/resolvers/auth/account.auth.resolver.ts b/services/module-accounts/resolvers/auth/account.auth.resolver.ts new file mode 100644 index 0000000..2d41d92 --- /dev/null +++ b/services/module-accounts/resolvers/auth/account.auth.resolver.ts @@ -0,0 +1,271 @@ +/* eslint-disable @typescript-eslint/explicit-function-return-type */ +/* eslint-disable @typescript-eslint/no-use-before-define */ +import { Arg, Ctx, Mutation, Query, Resolver } from 'type-graphql'; +import _ from 'lodash'; +import { v4 as uuid } from 'uuid'; + +import AccountModel from '@src/accounts/account.model'; +import { CodeConfirmInput, CodeRequestInput } from '@services/module-accounts/account.components'; +import Firebase from '@lib/seed/services/auth/FirebaseService'; +import { sendEmail } from '@lib/seed/services/email/EmailService'; +import { FirebaseTokenResult, SimpleResult } from '@lib/seed/interfaces/components'; +import moment from 'moment'; +import { generateCodeNumber } from '@seed/helpers/Utils'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { ApolloContext } from '@seed/interfaces/context'; +import { newError } from '@seed/helpers/Error'; +import { AvailableTranslation } from '@src/__components/components'; +import { DateTime } from 'luxon'; + +@Resolver(AccountModel) +export class AccountQueryCodeAuthResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + @Query(() => SimpleResult) + async logInMagicLink(@Arg('input') input: CodeRequestInput, @Ctx() ctx: ApolloContext): Promise { + try { + const magicLink = await Firebase.getInstance().getMagicLink(input.email); + await sendEmail('magicLinkLogIn', input.language || AvailableTranslation.en, input.email, { ...input, magicLink }); + return { message: 'sent' }; + } catch (error) { + throw error; + } + } + + @Query(() => SimpleResult) + async logInEmailCodeRequest(@Arg('input') input: CodeRequestInput): Promise { + const { email } = input; + + try { + // Check if exists on firebase already + const { accountCheck, model } = await getAccount(email); + + // Add the codes to the account + const code = await getCode(model, accountCheck); + await sendEmail('emailCodeLogIn', accountCheck.language, input.email, { firstname: accountCheck.firstName || 'stranger', code }); + return { message: 'sent' }; + } catch (error) { + throw error; + } + } + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + + */ + + @Mutation(() => FirebaseTokenResult) + async logInEmailCodeConfirm(@Arg('input') input: CodeConfirmInput): Promise { + const { code, email } = input; + + try { + // Check if exists on firebase already + const { accountCheck, model, firebaseUserData } = await getAccount(email); + // Get the codes + const codes = accountCheck.authCodes; + + // verify the code + const currentCode = _.find(codes, { code: code }); + if (!currentCode) throw newError(2300); + + // verify the expiration date + if (DateTime.utc() >= DateTime.fromJSDate(currentCode.expiresAt).toUTC()) throw newError(2301); + + // Get the token from firebase + const token = await Firebase.getInstance().createTokenId(email); + + // remove it from the array + await model.updateOneCustom( + { _id: firebaseUserData.uid }, + { + $pull: { authCodes: { code: code } }, + }, + null, + ); + return token; + } catch (error) { + throw error; + } + } +} + +@Resolver(AccountModel) +export class AccountMutationCodeAuthResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + + */ + + @Mutation(() => SimpleResult) + async signInMagicLink(@Arg('input') input: CodeRequestInput, @Ctx() ctx: ApolloContext): Promise { + try { + const { account } = await createAccount(input); + + const magicLink = await Firebase.getInstance().getMagicLink(input.email); + await sendEmail('magicLinkSignIn', account.language, account.email, { ...account, magicLink }); + return { message: 'sent' }; + } catch (error) { + throw error; + } + } + + @Mutation(() => SimpleResult) + async signInEmailCodeRequest(@Arg('input') input: CodeRequestInput): Promise { + // Check if exists on firebase already + try { + const { model, firebaseUserData, account } = await createAccount(input); + // Add the codes to the account + const code = await getCode(model, firebaseUserData); + await sendEmail('emailCodeSignIn', account.language, input.email, { firstname: account.firstName || 'stranger', code }); + return { message: 'sent' }; + } catch (error) { + throw error; + } + } +} + +async function getCode(model: AccountModel, data: AccountModel) { + // Get the codes + let codes = data.authCodes; + if (codes) { + // Remove the expired ones + _.remove(codes, (e) => { + const diffInMs = DateTime.fromJSDate(e.expiresAt).diffNow('milliseconds').milliseconds; + const MAX_TIMEOUT = 15 * 60 * 1000; + + if (diffInMs >= MAX_TIMEOUT) return true; + + return false; + }); + } else codes = []; + + // Check if too many attempt in the last 15 minutes + if (codes.length > 5) throw newError(2302); + + // Add the new one + const code = generateCodeNumber(); + + codes.push({ + code: code, + expiresAt: DateTime.utc() + .plus({ minutes: 15 }) + .toJSDate(), + }); + // Save in DB + await model.updateOneCustom( + { _id: data._id }, + { + $set: { authCodes: codes }, + }, + + null, + ); + return code; +} + +async function getAccount(email: string) { + let firebaseUserData; + try { + firebaseUserData = await Firebase.getInstance().getUserByEmail(email); + } catch (error) { + throw newError(2004, error); + } + + // Check if in DB or not + const model = new AccountModel(); + + const accountCheck = await (await model.db()).findOne({ _id: firebaseUserData.uid }); + // Verify if has thinkific signin + if (!accountCheck) throw newError(2004); + return { accountCheck, model, firebaseUserData }; +} + +async function createAccount(input: CodeRequestInput) { + let firebaseUserData; + try { + firebaseUserData = await Firebase.getInstance().getUserByEmail(input.email); + } catch (error) { + try { + firebaseUserData = await Firebase.getInstance().createUser({ + email: input.email, + password: uuid(), + }); + } catch (error) { + throw error; + } + } + + const dataToSave = { + ...input, + firebaseData: { + aud: firebaseUserData.aud, + emailVerified: firebaseUserData.email_verified, + firebase: JSON.stringify(firebaseUserData.firebase), + }, + updatedAt: new Date(), + }; + + // Check if in DB or not + const model = new AccountModel(); + const type = AccountTypeEnum.user; + + let account; + + const accountCheck = await (await model.db()).findOne({ _id: firebaseUserData.uid }); + // Verify if has thinkific signin + if (accountCheck) { + account = accountCheck; + await (await model.db()).updateOne( + { _id: firebaseUserData.uid }, + { + $set: { ...input }, + }, + ); + } else { + const types = [type]; + account = await model.saveOne( + { + ...dataToSave, + _id: firebaseUserData.uid, + types, + r: [AccountTypeEnum.admin, firebaseUserData.uid], + w: [AccountTypeEnum.admin, firebaseUserData.uid], + d: [AccountTypeEnum.admin, firebaseUserData.uid], + }, + null, + ); + } + return { model, firebaseUserData, account }; +} diff --git a/services/module-accounts/services/AccountService.ts b/services/module-accounts/services/AccountService.ts new file mode 100644 index 0000000..1c1deed --- /dev/null +++ b/services/module-accounts/services/AccountService.ts @@ -0,0 +1,72 @@ +import Firebase from '@seed/services/auth/FirebaseService'; +import { v4 as uuid } from 'uuid'; + +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { checkPermissions } from '@seed/graphql/AccessService'; +import { ApolloContext } from '@seed/interfaces/context'; +import { saveAccountToDB, saveAccountToDBAnonymous } from './account.service.helper'; +import FirebaseAuth from '@seed/services/auth/FirebaseAuthService'; +import { newError } from '@seed/helpers/Error'; + +export const signinGeneric = async (model: any, type: AccountTypeEnum, body: any, ctx: ApolloContext): Promise => { + try { + if (body.passwordConfirmation != body.password) throw newError(2006); + + const result = await saveAccountToDB(model, type, body, ctx); + + return result; + } catch (error) { + throw error; + } +}; + +export const signinGenericNoPassword = async (model: any, type: AccountTypeEnum, body: any, ctx: ApolloContext): Promise => { + try { + body.password = uuid(); + const result = await saveAccountToDB(model, type, body, ctx); + + return result; + } catch (error) { + throw error; + } +}; + +export const signinAnonymous = async (ctx: ApolloContext): Promise => { + try { + // Create the user in Firebase + const firebaseData = await FirebaseAuth.getInstance().loginAnonymous(); + // Save him in our database + await saveAccountToDBAnonymous(firebaseData, ctx); + + return firebaseData; + } catch (error) { + throw error; + } +}; + +export const deleteOneGeneric = async (model: any, type: AccountTypeEnum, id: string, ctx: ApolloContext): Promise => { + try { + const account = ctx.ctx.user; + + await model.getOne({ _id: id }, ctx); + checkPermissions(model, account, 'd'); + + // let firebaseUserData; + // Check if exists on firebase already + await Firebase.getInstance().deleteUser({ uid: id }); + + // // user exists + // if (firebaseUserGetResult.success) { + // firebaseUserData = firebaseUserGetResult.data; + // } else { + // throw firebaseUserGetResult.message; + // } + + const accountDataR = await (await model.db()).findOneAndDelete({ _id: id }); + if (accountDataR.ok == 0) throw newError(2007, accountDataR); + + return accountDataR.value; + } catch (error) { + throw error; + } +}; diff --git a/services/module-accounts/services/account.service.helper.ts b/services/module-accounts/services/account.service.helper.ts new file mode 100644 index 0000000..0cef7bb --- /dev/null +++ b/services/module-accounts/services/account.service.helper.ts @@ -0,0 +1,136 @@ +import { newError } from '@seed/helpers/Error'; +import { FirebaseTokenResult } from '@seed/interfaces/components'; +import { ApolloContext } from '@seed/interfaces/context'; +import Firebase from '@seed/services/auth/FirebaseService'; +import { createSetRequest } from '@seed/services/database/DBService'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import AccountModel from '@src/accounts/account.model'; +import { ApolloError } from 'apollo-server-lambda'; + +import { v4 as uuid } from 'uuid'; +import BaseAccountModel from '../account.model'; + +export const saveAccountToDB = async (model: T, type: AccountTypeEnum, body: any, ctx: ApolloContext): Promise => { + try { + const organisationId = ctx.ctx.organisationId; + + // Check if exists on firebase already + let firebaseUserData; + try { + firebaseUserData = await Firebase.getInstance().getUserByEmail(body.email); + } catch (error) { + try { + firebaseUserData = await Firebase.getInstance().createUser({ + email: body.email, + password: body.password, + }); + } catch (error) { + throw error; + } + } + + // Transform + delete body.password; + delete body.passwordConfirmation; + delete body.type; + + const dataToSave = { + ...body, + firebaseData: { + aud: firebaseUserData.aud, + emailVerified: firebaseUserData.email_verified, + firebase: JSON.stringify(firebaseUserData.firebase), + }, + updatedAt: new Date(), + }; + + // Check if in DB or not + let currentUserModel; + try { + currentUserModel = await model.getOne({ _id: firebaseUserData.uid }, null); + } catch (error) {} + + // If account exists -> Update this one + if (currentUserModel) { + // Check if already registered for that type + if (model.types.includes(type)) throw newError(2003); + + model.set(dataToSave); + model.types.push(type); + + if (organisationId) { + if (model.organisationIds) model.organisationIds.push(organisationId); + else model.organisationIds = [organisationId]; + } + + model.types = [...new Set(model.types)]; + model.organisationIds = [...new Set(model.organisationIds)]; + + return await model.updateOne( + { _id: model._id as any }, + { + ...model.get(), + }, + null, + ); + } + // If account doesn't exists -> Create this one + else { + const types = [type]; + if (organisationId) dataToSave.organisationIds = [organisationId]; + return await model.saveOne( + { + ...dataToSave, + _id: firebaseUserData.uid, + types, + r: [AccountTypeEnum.admin, firebaseUserData.uid], + w: [AccountTypeEnum.admin, firebaseUserData.uid], + d: [AccountTypeEnum.admin, firebaseUserData.uid], + }, + null, + ); + } + } catch (error) { + throw error; + } +}; + +export const saveAccountToDBAnonymous = async (tokenResult: FirebaseTokenResult, ctx: ApolloContext): Promise => { + try { + const model = new AccountModel(); + + const organisationId = ctx.ctx.organisationId; + + const addToSet: any = { + types: AccountTypeEnum.public, + }; + + if (organisationId) addToSet.organisationIds = organisationId; + + const dataToSave = { + email: 'guest-' + uuid() + '@firebase.com', + updatedAt: new Date(), + }; + + const accountDataR = await (await model.db()).findOneAndUpdate( + { _id: tokenResult.localId }, + { + $set: createSetRequest('', dataToSave), + $addToSet: addToSet, + $setOnInsert: { + _id: tokenResult.localId, + r: [AccountTypeEnum.admin, tokenResult.localId], + w: [AccountTypeEnum.admin, tokenResult.localId], + d: [AccountTypeEnum.admin, tokenResult.localId], + createdAt: new Date(), + }, + }, + { upsert: true, returnOriginal: false }, + ); + if (accountDataR.ok == 0) throw newError(2005, accountDataR); + + return accountDataR.value; + } catch (error) { + throw newError(2005, error); + } +}; diff --git a/services/module-booking/.gitignore b/services/module-booking/.gitignore new file mode 100644 index 0000000..9bea433 --- /dev/null +++ b/services/module-booking/.gitignore @@ -0,0 +1,2 @@ + +.DS_Store diff --git a/services/module-booking/README.md b/services/module-booking/README.md new file mode 100644 index 0000000..47a4941 --- /dev/null +++ b/services/module-booking/README.md @@ -0,0 +1,23 @@ +# module-booking + +## CONFIGURATION + +- Add the component in the project components +```` +export enum BookingsRessourceEnum { + workspaces = 'workspaces', +} +registerEnumType(BookingsRessourceEnum, { + name: 'BookingsRessourceEnum', +}); + +```` + +- Register resolver for the different ressources. In src create a bookings folder with a bookingFieldResolver + +- Import BookingEngineResolver and the field Revolver of your project + + + +## TODO +-- On edit verify ownership \ No newline at end of file diff --git a/services/module-booking/__tests/playground.gql b/services/module-booking/__tests/playground.gql new file mode 100644 index 0000000..536fc30 --- /dev/null +++ b/services/module-booking/__tests/playground.gql @@ -0,0 +1,11 @@ +mutation createBooking { + createOneBooking(input:{ressourceModel:workspaces,ressourceId:"718aa94e-f13d-4ee3-9610-c12c98750c08",dates:{startDate:"2020-11-01T23:15:17.044+0000",endDate:"2020-11-02T23:15:17.044+0000"},capacity:10}){_id,ownerId,owner{email}} +} + +query bookingsGetMany { + bookingsGetMany{_id} +} + +query bookingsGetOne { + bookingsGetOne(id:"2a3c6b3d-e8e5-4a4d-9c10-6e438ac22e10"){_id} +} \ No newline at end of file diff --git a/services/module-booking/components.errors.ts b/services/module-booking/components.errors.ts new file mode 100644 index 0000000..5b9a486 --- /dev/null +++ b/services/module-booking/components.errors.ts @@ -0,0 +1,6 @@ +export const engineBookingErrorConfig = { + /* Bookings error */ + 7000: 'That slot has already been reserved', + 7001: 'There was an error manipulating the dates', + 7002: 'That slot is not available ', +}; diff --git a/services/module-booking/functions/bookings/booking.service.ts b/services/module-booking/functions/bookings/booking.service.ts new file mode 100644 index 0000000..f70743d --- /dev/null +++ b/services/module-booking/functions/bookings/booking.service.ts @@ -0,0 +1,156 @@ +import { EngineModel } from '@seed/engine/EngineModel'; +import { BaseGraphModel } from '@seed/graphql/BaseGraphModel'; +import { newError } from '@seed/helpers/Error'; +import { converDateRangeToDateTime, isTimeSlotInDates, slotToDatesRange } from '@seed/helpers/Utils.dates'; +import { checkIfDatesAreInAvailabilities } from '@seed/helpers/Utils.dates.intervals'; +import { StatusEnum } from '@seed/interfaces/components'; +import { AvailabilityComponent, DateRangeComponent, SlotComponent } from '@seed/interfaces/components.dates'; +import { ApolloContext } from '@seed/interfaces/context'; +import { baseSearchFunction } from '@seed/services/database/DBRequestService'; +import { CheckoutEngineInput } from '@services/module-payments/functions/orders/schemas/order.schema.input'; +import { BookingArgsSchema, BookingNewInputSchema } from '@src/bookings/schemas/bookings.input'; +import { BookingSchema, BookingDBInterfaceSchema } from '@src/bookings/schemas/bookings.schema'; +import { BookingsRessourceEnum } from '@src/__components/components'; +import _ from 'lodash'; +import { concat } from 'lodash'; +import { DateTime } from 'luxon'; +import { DurationTypeEnum } from '../components'; +import BookingModel from './bookings.engine.model'; + +export interface NewOneBookingInterface { + model: T; + input: BookingNewInputSchema; + status: StatusEnum; + availabilities?: AvailabilityComponent; + ctx: ApolloContext; +} + +export interface SaveOneBookingInterface extends NewOneBookingInterface { + ressourceModel: BookingsRessourceEnum; + ressourceId: string; + rest: { + capacity: number; + comments?: string | undefined; + checkoutInfo?: CheckoutEngineInput | undefined; + durationType: DurationTypeEnum; + startToEnd?: DateRangeComponent | undefined; + slot?: SlotComponent | undefined; + }; + permissions: any; +} + +export class BookingService { + async getBookings(args: BookingArgsSchema, ctx?: ApolloContext | null): Promise { + const insi: any = { + model: new BookingModel() as any, + engine: true, + query: args, + ctx: ctx, + }; + + if (args.pagination) insi.pagination = args.pagination; + else insi.all = true; + + return baseSearchFunction(insi); + } + + async getBookingsToFilter(ressourceType: BookingsRessourceEnum, args: SlotComponent): Promise { + const idsToOmit: string[] = []; + + const bookings = await this.getBookings({ + dateRange: { + startDate: DateTime.fromJSDate(args.startDate).startOf('day').toJSDate(), + endDate: DateTime.fromJSDate(args.endDate).endOf('day').toJSDate(), + }, + }); + + if (bookings && bookings.length > 0) { + // Get the booking that are in those specifiq hours + bookings.forEach((element) => { + const bookingDates = converDateRangeToDateTime(element.dates); + if (isTimeSlotInDates(bookingDates, args.startTime || '00:00', args.endTime || '23:59')) { + const pIndex = _.findIndex(element.paths, { ressourceModel: ressourceType }); + if (pIndex !== -1) idsToOmit.push(element.paths[pIndex].ressourceId); + } + }); + } + + return _.uniq(idsToOmit); + } + + async createOneBookingWithEngine>(data: NewOneBookingInterface): Promise { + const { model, status, ctx, input, availabilities } = data; + const { ressourceModel, ressourceId, ...rest } = input; + + const permissions: any = { + r: [ctx.ctx.user._id, ...model.permissions.r], + w: [ctx.ctx.user._id, ...model.permissions.w], + d: [ctx.ctx.user._id, ...model.permissions.d], + }; + + return await this.saveBooking({ input, ressourceModel, ressourceId, rest, status, model, ctx, permissions, availabilities }); + } + + async createOneBookingWithModel(data: NewOneBookingInterface): Promise { + const { model, status, ctx, input, availabilities } = data; + const { ressourceModel, ressourceId, ...rest } = input; + + _.remove(model.r, function(n) { return n == "public";}) + _.remove(model.w, function(n) { return n == "public";}) + _.remove(model.d, function(n) { return n == "public";}) + + const permissions: any = { + r: [ctx.ctx.user._id, ...model.r], + w: [ctx.ctx.user._id, ...model.w], + d: [ctx.ctx.user._id, ...model.d], + }; + + + + return await this.saveBooking({ input, ressourceModel, ressourceId, rest, status, model, ctx, permissions, availabilities }); + } + + async saveBooking | BaseGraphModel>(data: SaveOneBookingInterface) { + const { ressourceModel, ressourceId, rest, status, model, ctx, permissions, input, availabilities } = data; + + const zone = process.env.TIMEZONE || 'utc'; + + const dates: DateRangeComponent[] = []; + let ressourceIdsToOmit: string[] = []; + + // (1) First check, in the bookings already made + + if (input.durationType == DurationTypeEnum.startToEnd && input.startToEnd) dates.push(input.startToEnd); + else if (input.durationType == DurationTypeEnum.slot && input.slot) dates.push(...slotToDatesRange(input.slot)); + + if (dates.length == 0) throw newError(7001); + + // (1) First check, in the availabilities + if (availabilities) { + const inAvailability = checkIfDatesAreInAvailabilities(converDateRangeToDateTime(dates, { zone }), availabilities); + if (!inAvailability) throw newError(7002); + } + + // (2) Second check + // Verify if no overlaps with others + if (input.slot) ressourceIdsToOmit = await this.getBookingsToFilter(ressourceModel, input.slot); + // Check if overlap + if (ressourceIdsToOmit.includes(ressourceId)) throw newError(7000); + + const newData: BookingDBInterfaceSchema = { + ...rest, + status, + dates, + paths: concat((model as any).getPath()), + ownerId: ctx.ctx.user._id, + }; + + return await new BookingModel(newData).saveOne({ + additionnalData: { + paths: concat((model as any).getPath()), + permissions, + }, + ctx, + }); + } +} diff --git a/services/module-booking/functions/bookings/bookings.engine.model.ts b/services/module-booking/functions/bookings/bookings.engine.model.ts new file mode 100644 index 0000000..0742fa0 --- /dev/null +++ b/services/module-booking/functions/bookings/bookings.engine.model.ts @@ -0,0 +1,71 @@ +import { ObjectType } from 'type-graphql'; +import { plainToClass } from 'class-transformer'; + +import { EngineModel } from '@lib/seed/engine/EngineModel'; +import { Permission } from '@seed/interfaces/permission'; +import { CustomSearchEngine } from '@seed/services/database/DBRequestService'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { IEngineSchema } from '@seed/engine/EngineSchema'; +import { ModelCollectionEnum } from '@src/__indexes/__collections'; +import { BookingDBInterfaceSchema, BookingDBSchema, BookingSchema } from '@src/bookings/schemas/bookings.schema'; +import { BookingArgsSchema } from '@src/bookings/schemas/bookings.input'; +import { RangeType } from '@seed/interfaces/components'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.admin], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class BookingModel extends EngineModel { + public constructor(input?: BookingDBInterfaceSchema & Partial) { + const dataInit = plainToClass(BookingDBSchema, input || {}); + super({ + // ...init, + collectionName: ModelCollectionEnum.bookings, + permissions: permissions, + dataInit, + }); + } + + plainToClass(plain: any): BookingSchema | BookingSchema[] { + return plainToClass(BookingSchema, plain); + } + + searchEngine(): CustomSearchEngine { + const sEngine: CustomSearchEngine = { + dateTab: { + operation: 'dateTab', + dateTabConfig: { + dbFromField: 'dates.startDate', + dbToField: 'dates.endDate', + }, + }, + dateRange: { + operation: 'dateRange', + dateRangeConfig: { + inputFromField: 'startDate', + inputToField: 'endDate', + dbFromField: 'dates.startDate', + dbToField: 'dates.endDate', + dateRangeType: RangeType.intersect, + }, + }, + ressourceId: { + operation: '$eq', + dbFName: 'paths.ressourceId', + }, + ressourceModel: { + operation: '$eq', + dbFName: 'paths.ressourceModel', + }, + status: { + operation: '$in', + }, + }; + + return sEngine; + } +} diff --git a/services/module-booking/functions/bookings/resolvers/bookings.engine.resolver.ts b/services/module-booking/functions/bookings/resolvers/bookings.engine.resolver.ts new file mode 100644 index 0000000..a6ebf4f --- /dev/null +++ b/services/module-booking/functions/bookings/resolvers/bookings.engine.resolver.ts @@ -0,0 +1,42 @@ +import { Resolver, Ctx, Mutation, Arg, Authorized } from 'type-graphql'; + +import BookingModel from '../bookings.engine.model'; +import { createEngineQueryResolver } from '@seed/engine/genericResolvers/BaseEngineResolver'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { BookingEngineArgsSchema, BookingEngineNewInputSchema } from '../schemas/bookings.engine.input'; +import { BookingEngineSchema } from '../schemas/bookings.engine.schema'; + +const BookingGenericQueryResolver = createEngineQueryResolver({ + domain: 'bookings', + schemaName: BookingEngineSchema, + modelName: BookingModel, + argsType: BookingEngineArgsSchema, + engineMiddleware: { + authorization: [AccountTypeEnum.user], + }, +}); + +@Resolver(BookingEngineSchema) +export class BookingEngineQueryResolver extends BookingGenericQueryResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ +} + +export class BookingEngineMutationResolver { + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ +} + +export const BookingEngineResolvers = [BookingEngineQueryResolver, BookingEngineMutationResolver]; diff --git a/services/module-booking/functions/bookings/schemas/bookings.engine.input.ts b/services/module-booking/functions/bookings/schemas/bookings.engine.input.ts new file mode 100644 index 0000000..acf2d87 --- /dev/null +++ b/services/module-booking/functions/bookings/schemas/bookings.engine.input.ts @@ -0,0 +1,43 @@ +import { Field, InputType, Int, ArgsType, ObjectType } from 'type-graphql'; + +import { GetManyArgsWithDateRange } from '@seed/graphql/Request'; +import { BookingEngineBaseSchema } from './bookings.engine.schema'; +import { IsRefExist } from '@seed/engine/decorators/db.guard'; +import { BookingsRessourceEnum } from '@src/__components/components'; +import { IsNotEmpty, ValidateIf, ValidateNested } from 'class-validator'; + +import { StatusEnum } from '@seed/interfaces/components'; + +@ObjectType() +@InputType() +export class BookingEngineNewInputSchema extends BookingEngineBaseSchema { + @Field(() => BookingsRessourceEnum) + @IsNotEmpty() + ressourceModel: BookingsRessourceEnum; + + @Field() + @IsNotEmpty() + @IsRefExist() + ressourceId: string; +} + +@InputType() +export class BookingEngineEditInputSchema { + @Field(() => Int, { nullable: true }) + capacity: number; + + @Field({ nullable: true }) + comments?: string; +} + +@ArgsType() +export class BookingEngineArgsSchema extends GetManyArgsWithDateRange { + @Field(() => BookingsRessourceEnum, { nullable: true }) + ressourceModel?: BookingsRessourceEnum; + + @Field({ nullable: true }) + ressourceId?: string; + + @Field(() => [StatusEnum], { nullable: true }) + status?: StatusEnum[]; +} diff --git a/services/module-booking/functions/bookings/schemas/bookings.engine.schema.ts b/services/module-booking/functions/bookings/schemas/bookings.engine.schema.ts new file mode 100644 index 0000000..09f1752 --- /dev/null +++ b/services/module-booking/functions/bookings/schemas/bookings.engine.schema.ts @@ -0,0 +1,76 @@ +import { IEngineSchema, MetaBy, MetaPermissions } from '@seed/engine/EngineSchema'; +import { EnginePathComponent, StatusEnum } from '@seed/interfaces/components'; +import { DateRangeComponent, SlotComponent } from '@seed/interfaces/components.dates'; +import { ApolloContext } from '@seed/interfaces/context'; +import AccountModel from '@src/accounts/account.model'; +import { Type } from 'class-transformer'; +import { ValidateNested, IsNotEmpty, ValidateIf } from 'class-validator'; +import { Ctx, Field, InputType, Int, ObjectType } from 'type-graphql'; +import { DurationTypeEnum } from '../../components'; + +@ObjectType() +@InputType() +export class BookingEngineBaseSchema { + @Field(() => DurationTypeEnum) + @IsNotEmpty() + durationType: DurationTypeEnum; + + @Field(() => DateRangeComponent, { nullable: true }) + @Type(() => DateRangeComponent) + @ValidateIf((o: BookingEngineBaseSchema) => { + return o.durationType == DurationTypeEnum.startToEnd; + }) + @IsNotEmpty() + @ValidateNested() + startToEnd?: DateRangeComponent; + + @Field(() => SlotComponent, { nullable: true }) + @Type(() => SlotComponent) + @ValidateIf((o: BookingEngineBaseSchema) => { + return o.durationType == DurationTypeEnum.slot; + }) + @IsNotEmpty() + @ValidateNested() + slot?: SlotComponent; + + @Field(() => Int, { nullable: true }) + capacity?: number; + + @Field({ nullable: true }) + comments?: string; +} + +@ObjectType() +export class BookingEngineDBInterfaceSchema extends BookingEngineBaseSchema { + @Field(() => StatusEnum) + status: StatusEnum; + + @Field(() => [EnginePathComponent]) + paths: EnginePathComponent[]; + + @Field() + @IsNotEmpty() + ownerId: string; + + @Field(() => [DateRangeComponent]) + @IsNotEmpty() + dates: DateRangeComponent[]; +} + +@ObjectType() +export class BookingEngineDBSchema extends BookingEngineDBInterfaceSchema { + @Field(() => AccountModel) + async owner(@Ctx() ctx: ApolloContext): Promise { + return ctx.ctx.loaders.accountLoader.load(this.ownerId); + } +} + +@ObjectType({ implements: IEngineSchema }) +export class BookingEngineSchema extends BookingEngineDBSchema implements IEngineSchema { + _id: string; + organisationId?: string | undefined; + by?: MetaBy | undefined; + permissions: MetaPermissions; + createdAt: Date; + updatedAt: Date; +} diff --git a/services/module-booking/functions/components.ts b/services/module-booking/functions/components.ts new file mode 100644 index 0000000..c4d648e --- /dev/null +++ b/services/module-booking/functions/components.ts @@ -0,0 +1,44 @@ +/* + ███████╗███╗ ██╗██╗ ██╗███╗ ███╗ + ██╔════╝████╗ ██║██║ ██║████╗ ████║ + █████╗ ██╔██╗ ██║██║ ██║██╔████╔██║ + ██╔══╝ ██║╚██╗██║██║ ██║██║╚██╔╝██║ + ███████╗██║ ╚████║╚██████╔╝██║ ╚═╝ ██║ + ╚══════╝╚═╝ ╚═══╝ ╚═════╝ ╚═╝ ╚═╝ +*/ + +import { IsRefExist } from '@seed/engine/decorators/db.guard'; +import { BookingsRessourceEnum } from '@src/__components/components'; +import { IsNotEmpty } from 'class-validator'; +import { Field, ObjectType, registerEnumType } from 'type-graphql'; + +export enum DurationTypeEnum { + startToEnd = 'startToEnd', + slot = 'slot', +} +registerEnumType(DurationTypeEnum, { + name: 'DurationTypeEnum', +}); + +/* + + ██████╗ ██████╗ ███╗ ███╗██████╗ ██████╗ ███╗ ██╗███████╗███╗ ██╗████████╗███████╗ +██╔════╝██╔═══██╗████╗ ████║██╔══██╗██╔═══██╗████╗ ██║██╔════╝████╗ ██║╚══██╔══╝██╔════╝ +██║ ██║ ██║██╔████╔██║██████╔╝██║ ██║██╔██╗ ██║█████╗ ██╔██╗ ██║ ██║ ███████╗ +██║ ██║ ██║██║╚██╔╝██║██╔═══╝ ██║ ██║██║╚██╗██║██╔══╝ ██║╚██╗██║ ██║ ╚════██║ +╚██████╗╚██████╔╝██║ ╚═╝ ██║██║ ╚██████╔╝██║ ╚████║███████╗██║ ╚████║ ██║ ███████║ + ╚═════╝ ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═════╝ ╚═╝ ╚═══╝╚══════╝╚═╝ ╚═══╝ ╚═╝ ╚══════╝ + +*/ + +@ObjectType() +export class GenericRefComponent { + @Field(() => BookingsRessourceEnum) + @IsNotEmpty() + ressourceModel: BookingsRessourceEnum; + + @Field() + @IsNotEmpty() + @IsRefExist() + ressourceId: string; +} diff --git a/services/module-booking/functions/crons/booking.cron.service.ts b/services/module-booking/functions/crons/booking.cron.service.ts new file mode 100644 index 0000000..a32fc02 --- /dev/null +++ b/services/module-booking/functions/crons/booking.cron.service.ts @@ -0,0 +1,31 @@ +import { StatusEnum } from "@seed/interfaces/components"; +import { SuccessResponse } from "@seed/interfaces/response"; +import DB from "@seed/services/database/DBService"; +import _ from "lodash"; +import { DateTime } from "luxon"; +import BookingModel from "../bookings/bookings.engine.model"; + +export class BookingCronService { + async removeDrafts(): Promise { + const bookings = await new BookingModel().getAll({ query:{ + 'createdAt' : {"$lte": DateTime.utc().minus({minutes:30}).toJSDate()}, + 'status': StatusEnum.draft, + }}) + + bookings.forEach((element,index) => { + bookings[index].status = StatusEnum.expired; + }); + + if (bookings.length > 0) { + // Move them to the other collection + await (await DB.getInstance()).db.collection('bookings.archives').insertMany(bookings); + + const bookingIds = _.map(bookings,'_id'); + await (await new BookingModel().db()).deleteMany({_id:{'$in':bookingIds}}); + + } + + return {success:true}; + } + +} diff --git a/services/module-booking/functions/helper.ts b/services/module-booking/functions/helper.ts new file mode 100644 index 0000000..55922dc --- /dev/null +++ b/services/module-booking/functions/helper.ts @@ -0,0 +1,63 @@ +import { getMinutesFromTimeInString, getNumberOfWeekendDays } from '@seed/helpers/Utils.dates'; +import { DateRangeComponent, SlotComponent } from '@seed/interfaces/components.dates'; +import { DateTime } from 'luxon'; + +export interface StartToEndFormat { + bookingDate: string; + bookingStart: string; + bookingEnd: string; + bookingDurationInMinutes: number; +} + +export interface SlotFormat { + bookingStart: string; + bookingEnd: string; + bookingHourStart: string; + bookingHourEnd: string; + bookingDurationInMinutes: number; +} + +export const formatStartToEnd = (input: DateRangeComponent, options?: { zone: string }): StartToEndFormat => { + const zone = options?.zone || process.env.TIMEZONE || 'utc'; + + const bookingStart = DateTime.fromJSDate(input.startDate, { zone }); + const bookingEnd = DateTime.fromJSDate(input.endDate, { zone }); + const durationObject = bookingEnd.diff(bookingStart, 'minutes').toObject(); + + return { + bookingDate: bookingStart.toFormat('dd/MM/yyyy'), + bookingStart: bookingStart.toFormat('HH:mm'), + bookingEnd: bookingEnd.toFormat('HH:mm'), + bookingDurationInMinutes: durationObject.minutes || 0, + }; +}; + +export const formatSlot = (input: SlotComponent, options?: { zone: string; removeWeekend?: boolean }): SlotFormat => { + const zone = options?.zone || process.env.TIMEZONE || 'utc'; + const removeWeekend = options?.removeWeekend || false; + const { startDate, endDate, startTime, endTime } = input; + + const bookingStart = DateTime.fromJSDate(startDate, { zone }); + let bookingEnd = DateTime.fromJSDate(endDate, { zone }); + const numberOfDay = bookingEnd.diff(bookingStart, 'days').toObject(); + + const startTimeInMin = getMinutesFromTimeInString(startTime); + const endTimeInMin = getMinutesFromTimeInString(endTime); + + let daysToRemove = 0; + if (removeWeekend) { + daysToRemove = getNumberOfWeekendDays(bookingStart, bookingEnd); + } + + const finalNumberOfDays = (numberOfDay.day || numberOfDay.days || 0) + 1 - daysToRemove; + + bookingEnd = bookingEnd.minus({ second: 1 }); + + return { + bookingStart: bookingStart.toFormat('dd/MM/yyyy'), + bookingEnd: bookingEnd.toFormat('dd/MM/yyyy'), + bookingHourStart: startTime, + bookingHourEnd: endTime, + bookingDurationInMinutes: finalNumberOfDays * (endTimeInMin - startTimeInMin), + }; +}; diff --git a/services/module-cms/.gitignore b/services/module-cms/.gitignore new file mode 100644 index 0000000..9bea433 --- /dev/null +++ b/services/module-cms/.gitignore @@ -0,0 +1,2 @@ + +.DS_Store diff --git a/services/module-cms/README.md b/services/module-cms/README.md new file mode 100644 index 0000000..710a2c8 --- /dev/null +++ b/services/module-cms/README.md @@ -0,0 +1,7 @@ +# Make-it CMS Service + +[![](https://ci6.googleusercontent.com/proxy/2HHmNs0zn0uGJb9RqML73pX6Bmd-BntMaIYR6IpXeIHKqn2_tG60C1pLNtMHoLHOk1I5CT8k5x9gvb-C1zuudHnD5HDOpvHljVYcKmDL--rb0Ar8IT4UB7YaG1c=s0-d-e1-ft#https://makeit-assets.s3.eu-central-1.amazonaws.com/signature_sanawar.png)](https://makeit-studio.com) + +![Build Status](https://travis-ci.org/joemccann/dillinger.svg?branch=master) + +### This is a private repo diff --git a/services/module-cms/__tests/pages.playground.graphql b/services/module-cms/__tests/pages.playground.graphql new file mode 100644 index 0000000..bce4c15 --- /dev/null +++ b/services/module-cms/__tests/pages.playground.graphql @@ -0,0 +1,31 @@ +query pages { + pages { + _id + title + content + } +} + +mutation addOnePage { + pagesAddOne(input: { title: "test New Title" }) { + _id + title + content + } +} + +mutation editOnePage { + pagesEditOne(id: "b59b36db-7f2c-44ab-8793-b47f773a2d2e", input: { title: "Updated Title" }) { + _id + title + content + } +} + +mutation deleteOnePage { + pagesDeleteOne(id: "b59b36db-7f2c-44ab-8793-b47f773a2d2e") { + _id + title + content + } +} diff --git a/services/module-cms/__tests/project-settings.playground.graphql b/services/module-cms/__tests/project-settings.playground.graphql new file mode 100644 index 0000000..54c8d73 --- /dev/null +++ b/services/module-cms/__tests/project-settings.playground.graphql @@ -0,0 +1 @@ +mutation projectSettingsAddOne {projectSettingsAddOne(input:{key:"FILE_allowedMime",value:"image/jpeg,image/png,image/gif,application/pdf,image/svg+xml,application/xml,application/zip"}){_id}} \ No newline at end of file diff --git a/services/module-cms/components.ts b/services/module-cms/components.ts new file mode 100644 index 0000000..7dfde2a --- /dev/null +++ b/services/module-cms/components.ts @@ -0,0 +1,33 @@ +import { registerEnumType, ObjectType, Field, InputType } from 'type-graphql'; +import { TranslatableComponent } from '@src/__components/components'; + +/* + ███████╗███╗ ██╗██╗ ██╗███╗ ███╗ + ██╔════╝████╗ ██║██║ ██║████╗ ████║ + █████╗ ██╔██╗ ██║██║ ██║██╔████╔██║ + ██╔══╝ ██║╚██╗██║██║ ██║██║╚██╔╝██║ + ███████╗██║ ╚████║╚██████╔╝██║ ╚═╝ ██║ + ╚══════╝╚═╝ ╚═══╝ ╚═════╝ ╚═╝ ╚═╝ +*/ + +/* + + ██████╗ ██████╗ ███╗ ███╗██████╗ ██████╗ ███╗ ██╗███████╗███╗ ██╗████████╗███████╗ +██╔════╝██╔═══██╗████╗ ████║██╔══██╗██╔═══██╗████╗ ██║██╔════╝████╗ ██║╚══██╔══╝██╔════╝ +██║ ██║ ██║██╔████╔██║██████╔╝██║ ██║██╔██╗ ██║█████╗ ██╔██╗ ██║ ██║ ███████╗ +██║ ██║ ██║██║╚██╔╝██║██╔═══╝ ██║ ██║██║╚██╗██║██╔══╝ ██║╚██╗██║ ██║ ╚════██║ +╚██████╗╚██████╔╝██║ ╚═╝ ██║██║ ╚██████╔╝██║ ╚████║███████╗██║ ╚████║ ██║ ███████║ + ╚═════╝ ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═════╝ ╚═╝ ╚═══╝╚══════╝╚═╝ ╚═══╝ ╚═╝ ╚══════╝ + +*/ + +@ObjectType() +@InputType('SEOEventInput') +export class SEOEvent { + @Field() + title: TranslatableComponent; + @Field() + description: TranslatableComponent; + @Field() + keywords: TranslatableComponent; +} diff --git a/services/module-cms/functions/categories/category.model.ts b/services/module-cms/functions/categories/category.model.ts new file mode 100644 index 0000000..62a6cd5 --- /dev/null +++ b/services/module-cms/functions/categories/category.model.ts @@ -0,0 +1,80 @@ +import { ObjectType, Field, ID, ArgsType, InputType, Int } from 'type-graphql'; + +import { GetArgs } from '@seed/graphql/Request'; + +import { TranslatableComponent, RessourceEnum } from '@src/__components/components'; + +import { Permission } from '@seed/interfaces/permission'; +import BaseSEOModel from '@seed/graphql/baseModels/BaseSeoModel'; +import { AccountTypeEnum } from '@src/accounts/account.components'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.public], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class CategoryModel extends BaseSEOModel { + public constructor() { + super('categories', permissions); + } + // eslint-disable-next-line @typescript-eslint/no-unused-vars + @Field(() => ID) + readonly _id: string; + + @Field(() => RessourceEnum) + ressourceType: RessourceEnum; + + @Field(() => TranslatableComponent) + title: TranslatableComponent; + + @Field(() => TranslatableComponent, { nullable: true }) + content: TranslatableComponent; + + @Field({ nullable: true }) + colorCode?: string; + + searchOptions(): string[] { + return ['title']; + } +} + +@ArgsType() +export class CategoryArgs { + @Field({ nullable: true }) + search?: string; + + @Field(() => RessourceEnum) + ressourceType: RessourceEnum; + + @Field(() => GetArgs, { nullable: true }) + pagination?: GetArgs; +} + +@InputType() +export class NewCategoryInput implements Partial { + @Field(() => RessourceEnum) + ressourceType: RessourceEnum; + + @Field(() => TranslatableComponent) + title: TranslatableComponent; + @Field(() => TranslatableComponent, { nullable: true }) + content: TranslatableComponent; + @Field({ nullable: true }) + colorCode?: string; +} + +@InputType() +export class EditCategoryInput implements Partial { + @Field(() => RessourceEnum) + ressourceType: RessourceEnum; + + @Field(() => TranslatableComponent) + title: TranslatableComponent; + @Field(() => TranslatableComponent, { nullable: true }) + content: TranslatableComponent; + @Field({ nullable: true }) + colorCode?: string; +} diff --git a/services/module-cms/functions/categories/category.resolver.admin.ts b/services/module-cms/functions/categories/category.resolver.admin.ts new file mode 100644 index 0000000..1bb00e0 --- /dev/null +++ b/services/module-cms/functions/categories/category.resolver.admin.ts @@ -0,0 +1,28 @@ +import { Resolver, Query, FieldResolver, Root, Arg, Mutation, Ctx, Args, Authorized, ResolverInterface } from 'type-graphql'; +import { createBaseResolver } from '@seed/graphql/baseResolvers/BaseResolver'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import CategoryModel, { CategoryArgs, NewCategoryInput, EditCategoryInput } from '@services/module-cms/functions/categories/category.model'; + +const CategoryAdminBaseResolver = createBaseResolver('categories', CategoryModel, CategoryArgs, NewCategoryInput, EditCategoryInput, [ + AccountTypeEnum.admin, +]); + +@Resolver(CategoryModel) +export default class CategoryAdminResolver extends CategoryAdminBaseResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ +} diff --git a/services/module-cms/functions/categories/category.resolver.fields.ts b/services/module-cms/functions/categories/category.resolver.fields.ts new file mode 100644 index 0000000..05380ab --- /dev/null +++ b/services/module-cms/functions/categories/category.resolver.fields.ts @@ -0,0 +1,15 @@ +import { Resolver } from 'type-graphql'; + +import CategoryModel from '@services/module-cms/functions/categories/category.model'; + +@Resolver(CategoryModel) +export default class CategoryFieldResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ +} diff --git a/services/module-cms/functions/categories/category.resolver.ts b/services/module-cms/functions/categories/category.resolver.ts new file mode 100644 index 0000000..65bcff6 --- /dev/null +++ b/services/module-cms/functions/categories/category.resolver.ts @@ -0,0 +1,26 @@ +import { Resolver, Query, FieldResolver, Root, Arg, Mutation, Ctx, Args, Authorized, ResolverInterface } from 'type-graphql'; + +import CategoryModel, { CategoryArgs } from '@services/module-cms/functions/categories/category.model'; +import { createBasePublicResolver } from '@seed/graphql/baseResolvers/BasePublicResolver'; + +const CategoryBaseResolver = createBasePublicResolver('categories', CategoryModel, CategoryArgs); + +@Resolver(CategoryModel) +export default class CategoryResolver extends CategoryBaseResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ +} diff --git a/services/module-cms/functions/countries/countries.json b/services/module-cms/functions/countries/countries.json new file mode 100644 index 0000000..a25c983 --- /dev/null +++ b/services/module-cms/functions/countries/countries.json @@ -0,0 +1,1254 @@ +{ + "countries": [ + { + "countryCode": "AD", + "countryName": "Andorra", + "currencyCode": "EUR" + }, + { + "countryCode": "AE", + "countryName": "United Arab Emirates", + "currencyCode": "AED" + }, + { + "countryCode": "AF", + "countryName": "Afghanistan", + "currencyCode": "AFN" + }, + { + "countryCode": "AG", + "countryName": "Antigua and Barbuda", + "currencyCode": "XCD" + }, + { + "countryCode": "AI", + "countryName": "Anguilla", + "currencyCode": "XCD" + }, + { + "countryCode": "AL", + "countryName": "Albania", + "currencyCode": "ALL" + }, + { + "countryCode": "AM", + "countryName": "Armenia", + "currencyCode": "AMD" + }, + { + "countryCode": "AO", + "countryName": "Angola", + "currencyCode": "AOA" + }, + { + "countryCode": "AQ", + "countryName": "Antarctica", + "currencyCode": "" + }, + { + "countryCode": "AR", + "countryName": "Argentina", + "currencyCode": "ARS" + }, + { + "countryCode": "AS", + "countryName": "American Samoa", + "currencyCode": "USD" + }, + { + "countryCode": "AT", + "countryName": "Austria", + "currencyCode": "EUR" + }, + { + "countryCode": "AU", + "countryName": "Australia", + "currencyCode": "AUD" + }, + { + "countryCode": "AW", + "countryName": "Aruba", + "currencyCode": "AWG" + }, + { + "countryCode": "AX", + "countryName": "Åland", + "currencyCode": "EUR" + }, + { + "countryCode": "AZ", + "countryName": "Azerbaijan", + "currencyCode": "AZN" + }, + { + "countryCode": "BA", + "countryName": "Bosnia and Herzegovina", + "currencyCode": "BAM" + }, + { + "countryCode": "BB", + "countryName": "Barbados", + "currencyCode": "BBD" + }, + { + "countryCode": "BD", + "countryName": "Bangladesh", + "currencyCode": "BDT" + }, + { + "countryCode": "BE", + "countryName": "Belgium", + "currencyCode": "EUR" + }, + { + "countryCode": "BF", + "countryName": "Burkina Faso", + "currencyCode": "XOF" + }, + { + "countryCode": "BG", + "countryName": "Bulgaria", + "currencyCode": "BGN" + }, + { + "countryCode": "BH", + "countryName": "Bahrain", + "currencyCode": "BHD" + }, + { + "countryCode": "BI", + "countryName": "Burundi", + "currencyCode": "BIF" + }, + { + "countryCode": "BJ", + "countryName": "Benin", + "currencyCode": "XOF" + }, + { + "countryCode": "BL", + "countryName": "Saint Barthélemy", + "currencyCode": "EUR" + }, + { + "countryCode": "BM", + "countryName": "Bermuda", + "currencyCode": "BMD" + }, + { + "countryCode": "BN", + "countryName": "Brunei", + "currencyCode": "BND" + }, + { + "countryCode": "BO", + "countryName": "Bolivia", + "currencyCode": "BOB" + }, + { + "countryCode": "BQ", + "countryName": "Bonaire", + "currencyCode": "USD" + }, + { + "countryCode": "BR", + "countryName": "Brazil", + "currencyCode": "BRL" + }, + { + "countryCode": "BS", + "countryName": "Bahamas", + "currencyCode": "BSD" + }, + { + "countryCode": "BT", + "countryName": "Bhutan", + "currencyCode": "BTN" + }, + { + "countryCode": "BV", + "countryName": "Bouvet Island", + "currencyCode": "NOK" + }, + { + "countryCode": "BW", + "countryName": "Botswana", + "currencyCode": "BWP" + }, + { + "countryCode": "BY", + "countryName": "Belarus", + "currencyCode": "BYR" + }, + { + "countryCode": "BZ", + "countryName": "Belize", + "currencyCode": "BZD" + }, + { + "countryCode": "CA", + "countryName": "Canada", + "currencyCode": "CAD" + }, + { + "countryCode": "CC", + "countryName": "Cocos [Keeling] Islands", + "currencyCode": "AUD" + }, + { + "countryCode": "CD", + "countryName": "Democratic Republic of the Congo", + "currencyCode": "CDF" + }, + { + "countryCode": "CF", + "countryName": "Central African Republic", + "currencyCode": "XAF" + }, + { + "countryCode": "CG", + "countryName": "Republic of the Congo", + "currencyCode": "XAF" + }, + { + "countryCode": "CH", + "countryName": "Switzerland", + "currencyCode": "CHF" + }, + { + "countryCode": "CI", + "countryName": "Ivory Coast", + "currencyCode": "XOF" + }, + { + "countryCode": "CK", + "countryName": "Cook Islands", + "currencyCode": "NZD" + }, + { + "countryCode": "CL", + "countryName": "Chile", + "currencyCode": "CLP" + }, + { + "countryCode": "CM", + "countryName": "Cameroon", + "currencyCode": "XAF" + }, + { + "countryCode": "CN", + "countryName": "China", + "currencyCode": "CNY" + }, + { + "countryCode": "CO", + "countryName": "Colombia", + "currencyCode": "COP" + }, + { + "countryCode": "CR", + "countryName": "Costa Rica", + "currencyCode": "CRC" + }, + { + "countryCode": "CU", + "countryName": "Cuba", + "currencyCode": "CUP" + }, + { + "countryCode": "CV", + "countryName": "Cape Verde", + "currencyCode": "CVE" + }, + { + "countryCode": "CW", + "countryName": "Curacao", + "currencyCode": "ANG" + }, + { + "countryCode": "CX", + "countryName": "Christmas Island", + "currencyCode": "AUD" + }, + { + "countryCode": "CY", + "countryName": "Cyprus", + "currencyCode": "EUR" + }, + { + "countryCode": "CZ", + "countryName": "Czechia", + "currencyCode": "CZK" + }, + { + "countryCode": "DE", + "countryName": "Germany", + "currencyCode": "EUR" + }, + { + "countryCode": "DJ", + "countryName": "Djibouti", + "currencyCode": "DJF" + }, + { + "countryCode": "DK", + "countryName": "Denmark", + "currencyCode": "DKK" + }, + { + "countryCode": "DM", + "countryName": "Dominica", + "currencyCode": "XCD" + }, + { + "countryCode": "DO", + "countryName": "Dominican Republic", + "currencyCode": "DOP" + }, + { + "countryCode": "DZ", + "countryName": "Algeria", + "currencyCode": "DZD" + }, + { + "countryCode": "EC", + "countryName": "Ecuador", + "currencyCode": "USD" + }, + { + "countryCode": "EE", + "countryName": "Estonia", + "currencyCode": "EUR" + }, + { + "countryCode": "EG", + "countryName": "Egypt", + "currencyCode": "EGP" + }, + { + "countryCode": "EH", + "countryName": "Western Sahara", + "currencyCode": "MAD" + }, + { + "countryCode": "ER", + "countryName": "Eritrea", + "currencyCode": "ERN" + }, + { + "countryCode": "ES", + "countryName": "Spain", + "currencyCode": "EUR" + }, + { + "countryCode": "ET", + "countryName": "Ethiopia", + "currencyCode": "ETB" + }, + { + "countryCode": "FI", + "countryName": "Finland", + "currencyCode": "EUR" + }, + { + "countryCode": "FJ", + "countryName": "Fiji", + "currencyCode": "FJD" + }, + { + "countryCode": "FK", + "countryName": "Falkland Islands", + "currencyCode": "FKP" + }, + { + "countryCode": "FM", + "countryName": "Micronesia", + "currencyCode": "USD" + }, + { + "countryCode": "FO", + "countryName": "Faroe Islands", + "currencyCode": "DKK" + }, + { + "countryCode": "FR", + "countryName": "France", + "currencyCode": "EUR" + }, + { + "countryCode": "GA", + "countryName": "Gabon", + "currencyCode": "XAF" + }, + { + "countryCode": "GB", + "countryName": "United Kingdom", + "currencyCode": "GBP" + }, + { + "countryCode": "GD", + "countryName": "Grenada", + "currencyCode": "XCD" + }, + { + "countryCode": "GE", + "countryName": "Georgia", + "currencyCode": "GEL" + }, + { + "countryCode": "GF", + "countryName": "French Guiana", + "currencyCode": "EUR" + }, + { + "countryCode": "GG", + "countryName": "Guernsey", + "currencyCode": "GBP" + }, + { + "countryCode": "GH", + "countryName": "Ghana", + "currencyCode": "GHS" + }, + { + "countryCode": "GI", + "countryName": "Gibraltar", + "currencyCode": "GIP" + }, + { + "countryCode": "GL", + "countryName": "Greenland", + "currencyCode": "DKK" + }, + { + "countryCode": "GM", + "countryName": "Gambia", + "currencyCode": "GMD" + }, + { + "countryCode": "GN", + "countryName": "Guinea", + "currencyCode": "GNF" + }, + { + "countryCode": "GP", + "countryName": "Guadeloupe", + "currencyCode": "EUR" + }, + { + "countryCode": "GQ", + "countryName": "Equatorial Guinea", + "currencyCode": "XAF" + }, + { + "countryCode": "GR", + "countryName": "Greece", + "currencyCode": "EUR" + }, + { + "countryCode": "GS", + "countryName": "South Georgia and the South Sandwich Islands", + "currencyCode": "GBP" + }, + { + "countryCode": "GT", + "countryName": "Guatemala", + "currencyCode": "GTQ" + }, + { + "countryCode": "GU", + "countryName": "Guam", + "currencyCode": "USD" + }, + { + "countryCode": "GW", + "countryName": "Guinea-Bissau", + "currencyCode": "XOF" + }, + { + "countryCode": "GY", + "countryName": "Guyana", + "currencyCode": "GYD" + }, + { + "countryCode": "HK", + "countryName": "Hong Kong", + "currencyCode": "HKD" + }, + { + "countryCode": "HM", + "countryName": "Heard Island and McDonald Islands", + "currencyCode": "AUD" + }, + { + "countryCode": "HN", + "countryName": "Honduras", + "currencyCode": "HNL" + }, + { + "countryCode": "HR", + "countryName": "Croatia", + "currencyCode": "HRK" + }, + { + "countryCode": "HT", + "countryName": "Haiti", + "currencyCode": "HTG" + }, + { + "countryCode": "HU", + "countryName": "Hungary", + "currencyCode": "HUF" + }, + { + "countryCode": "ID", + "countryName": "Indonesia", + "currencyCode": "IDR" + }, + { + "countryCode": "IE", + "countryName": "Ireland", + "currencyCode": "EUR" + }, + { + "countryCode": "IL", + "countryName": "Israel", + "currencyCode": "ILS" + }, + { + "countryCode": "IM", + "countryName": "Isle of Man", + "currencyCode": "GBP" + }, + { + "countryCode": "IN", + "countryName": "India", + "currencyCode": "INR" + }, + { + "countryCode": "IO", + "countryName": "British Indian Ocean Territory", + "currencyCode": "USD" + }, + { + "countryCode": "IQ", + "countryName": "Iraq", + "currencyCode": "IQD" + }, + { + "countryCode": "IR", + "countryName": "Iran", + "currencyCode": "IRR" + }, + { + "countryCode": "IS", + "countryName": "Iceland", + "currencyCode": "ISK" + }, + { + "countryCode": "IT", + "countryName": "Italy", + "currencyCode": "EUR" + }, + { + "countryCode": "JE", + "countryName": "Jersey", + "currencyCode": "GBP" + }, + { + "countryCode": "JM", + "countryName": "Jamaica", + "currencyCode": "JMD" + }, + { + "countryCode": "JO", + "countryName": "Jordan", + "currencyCode": "JOD" + }, + { + "countryCode": "JP", + "countryName": "Japan", + "currencyCode": "JPY" + }, + { + "countryCode": "KE", + "countryName": "Kenya", + "currencyCode": "KES" + }, + { + "countryCode": "KG", + "countryName": "Kyrgyzstan", + "currencyCode": "KGS" + }, + { + "countryCode": "KH", + "countryName": "Cambodia", + "currencyCode": "KHR" + }, + { + "countryCode": "KI", + "countryName": "Kiribati", + "currencyCode": "AUD" + }, + { + "countryCode": "KM", + "countryName": "Comoros", + "currencyCode": "KMF" + }, + { + "countryCode": "KN", + "countryName": "Saint Kitts and Nevis", + "currencyCode": "XCD" + }, + { + "countryCode": "KP", + "countryName": "North Korea", + "currencyCode": "KPW" + }, + { + "countryCode": "KR", + "countryName": "South Korea", + "currencyCode": "KRW" + }, + { + "countryCode": "KW", + "countryName": "Kuwait", + "currencyCode": "KWD" + }, + { + "countryCode": "KY", + "countryName": "Cayman Islands", + "currencyCode": "KYD" + }, + { + "countryCode": "KZ", + "countryName": "Kazakhstan", + "currencyCode": "KZT" + }, + { + "countryCode": "LA", + "countryName": "Laos", + "currencyCode": "LAK" + }, + { + "countryCode": "LB", + "countryName": "Lebanon", + "currencyCode": "LBP" + }, + { + "countryCode": "LC", + "countryName": "Saint Lucia", + "currencyCode": "XCD" + }, + { + "countryCode": "LI", + "countryName": "Liechtenstein", + "currencyCode": "CHF" + }, + { + "countryCode": "LK", + "countryName": "Sri Lanka", + "currencyCode": "LKR" + }, + { + "countryCode": "LR", + "countryName": "Liberia", + "currencyCode": "LRD" + }, + { + "countryCode": "LS", + "countryName": "Lesotho", + "currencyCode": "LSL" + }, + { + "countryCode": "LT", + "countryName": "Lithuania", + "currencyCode": "EUR" + }, + { + "countryCode": "LU", + "countryName": "Luxembourg", + "currencyCode": "EUR" + }, + { + "countryCode": "LV", + "countryName": "Latvia", + "currencyCode": "EUR" + }, + { + "countryCode": "LY", + "countryName": "Libya", + "currencyCode": "LYD" + }, + { + "countryCode": "MA", + "countryName": "Morocco", + "currencyCode": "MAD" + }, + { + "countryCode": "MC", + "countryName": "Monaco", + "currencyCode": "EUR" + }, + { + "countryCode": "MD", + "countryName": "Moldova", + "currencyCode": "MDL" + }, + { + "countryCode": "ME", + "countryName": "Montenegro", + "currencyCode": "EUR" + }, + { + "countryCode": "MF", + "countryName": "Saint Martin", + "currencyCode": "EUR" + }, + { + "countryCode": "MG", + "countryName": "Madagascar", + "currencyCode": "MGA" + }, + { + "countryCode": "MH", + "countryName": "Marshall Islands", + "currencyCode": "USD" + }, + { + "countryCode": "MK", + "countryName": "Macedonia", + "currencyCode": "MKD" + }, + { + "countryCode": "ML", + "countryName": "Mali", + "currencyCode": "XOF" + }, + { + "countryCode": "MM", + "countryName": "Myanmar [Burma]", + "currencyCode": "MMK" + }, + { + "countryCode": "MN", + "countryName": "Mongolia", + "currencyCode": "MNT" + }, + { + "countryCode": "MO", + "countryName": "Macao", + "currencyCode": "MOP" + }, + { + "countryCode": "MP", + "countryName": "Northern Mariana Islands", + "currencyCode": "USD" + }, + { + "countryCode": "MQ", + "countryName": "Martinique", + "currencyCode": "EUR" + }, + { + "countryCode": "MR", + "countryName": "Mauritania", + "currencyCode": "MRO" + }, + { + "countryCode": "MS", + "countryName": "Montserrat", + "currencyCode": "XCD" + }, + { + "countryCode": "MT", + "countryName": "Malta", + "currencyCode": "EUR" + }, + { + "countryCode": "MU", + "countryName": "Mauritius", + "currencyCode": "MUR" + }, + { + "countryCode": "MV", + "countryName": "Maldives", + "currencyCode": "MVR" + }, + { + "countryCode": "MW", + "countryName": "Malawi", + "currencyCode": "MWK" + }, + { + "countryCode": "MX", + "countryName": "Mexico", + "currencyCode": "MXN" + }, + { + "countryCode": "MY", + "countryName": "Malaysia", + "currencyCode": "MYR" + }, + { + "countryCode": "MZ", + "countryName": "Mozambique", + "currencyCode": "MZN" + }, + { + "countryCode": "NA", + "countryName": "Namibia", + "currencyCode": "NAD" + }, + { + "countryCode": "NC", + "countryName": "New Caledonia", + "currencyCode": "XPF" + }, + { + "countryCode": "NE", + "countryName": "Niger", + "currencyCode": "XOF" + }, + { + "countryCode": "NF", + "countryName": "Norfolk Island", + "currencyCode": "AUD" + }, + { + "countryCode": "NG", + "countryName": "Nigeria", + "currencyCode": "NGN" + }, + { + "countryCode": "NI", + "countryName": "Nicaragua", + "currencyCode": "NIO" + }, + { + "countryCode": "NL", + "countryName": "Netherlands", + "currencyCode": "EUR" + }, + { + "countryCode": "NO", + "countryName": "Norway", + "currencyCode": "NOK" + }, + { + "countryCode": "NP", + "countryName": "Nepal", + "currencyCode": "NPR" + }, + { + "countryCode": "NR", + "countryName": "Nauru", + "currencyCode": "AUD" + }, + { + "countryCode": "NU", + "countryName": "Niue", + "currencyCode": "NZD" + }, + { + "countryCode": "NZ", + "countryName": "New Zealand", + "currencyCode": "NZD" + }, + { + "countryCode": "OM", + "countryName": "Oman", + "currencyCode": "OMR" + }, + { + "countryCode": "PA", + "countryName": "Panama", + "currencyCode": "PAB" + }, + { + "countryCode": "PE", + "countryName": "Peru", + "currencyCode": "PEN" + }, + { + "countryCode": "PF", + "countryName": "French Polynesia", + "currencyCode": "XPF" + }, + { + "countryCode": "PG", + "countryName": "Papua New Guinea", + "currencyCode": "PGK" + }, + { + "countryCode": "PH", + "countryName": "Philippines", + "currencyCode": "PHP" + }, + { + "countryCode": "PK", + "countryName": "Pakistan", + "currencyCode": "PKR" + }, + { + "countryCode": "PL", + "countryName": "Poland", + "currencyCode": "PLN" + }, + { + "countryCode": "PM", + "countryName": "Saint Pierre and Miquelon", + "currencyCode": "EUR" + }, + { + "countryCode": "PN", + "countryName": "Pitcairn Islands", + "currencyCode": "NZD" + }, + { + "countryCode": "PR", + "countryName": "Puerto Rico", + "currencyCode": "USD" + }, + { + "countryCode": "PS", + "countryName": "Palestine", + "currencyCode": "ILS" + }, + { + "countryCode": "PT", + "countryName": "Portugal", + "currencyCode": "EUR" + }, + { + "countryCode": "PW", + "countryName": "Palau", + "currencyCode": "USD" + }, + { + "countryCode": "PY", + "countryName": "Paraguay", + "currencyCode": "PYG" + }, + { + "countryCode": "QA", + "countryName": "Qatar", + "currencyCode": "QAR" + }, + { + "countryCode": "RE", + "countryName": "Réunion", + "currencyCode": "EUR" + }, + { + "countryCode": "RO", + "countryName": "Romania", + "currencyCode": "RON" + }, + { + "countryCode": "RS", + "countryName": "Serbia", + "currencyCode": "RSD" + }, + { + "countryCode": "RU", + "countryName": "Russia", + "currencyCode": "RUB" + }, + { + "countryCode": "RW", + "countryName": "Rwanda", + "currencyCode": "RWF" + }, + { + "countryCode": "SA", + "countryName": "Saudi Arabia", + "currencyCode": "SAR" + }, + { + "countryCode": "SB", + "countryName": "Solomon Islands", + "currencyCode": "SBD" + }, + { + "countryCode": "SC", + "countryName": "Seychelles", + "currencyCode": "SCR" + }, + { + "countryCode": "SD", + "countryName": "Sudan", + "currencyCode": "SDG" + }, + { + "countryCode": "SE", + "countryName": "Sweden", + "currencyCode": "SEK" + }, + { + "countryCode": "SG", + "countryName": "Singapore", + "currencyCode": "SGD" + }, + { + "countryCode": "SH", + "countryName": "Saint Helena", + "currencyCode": "SHP" + }, + { + "countryCode": "SI", + "countryName": "Slovenia", + "currencyCode": "EUR" + }, + { + "countryCode": "SJ", + "countryName": "Svalbard and Jan Mayen", + "currencyCode": "NOK" + }, + { + "countryCode": "SK", + "countryName": "Slovakia", + "currencyCode": "EUR" + }, + { + "countryCode": "SL", + "countryName": "Sierra Leone", + "currencyCode": "SLL" + }, + { + "countryCode": "SM", + "countryName": "San Marino", + "currencyCode": "EUR" + }, + { + "countryCode": "SN", + "countryName": "Senegal", + "currencyCode": "XOF" + }, + { + "countryCode": "SO", + "countryName": "Somalia", + "currencyCode": "SOS" + }, + { + "countryCode": "SR", + "countryName": "Suriname", + "currencyCode": "SRD" + }, + { + "countryCode": "SS", + "countryName": "South Sudan", + "currencyCode": "SSP" + }, + { + "countryCode": "ST", + "countryName": "São Tomé and Príncipe", + "currencyCode": "STD" + }, + { + "countryCode": "SV", + "countryName": "El Salvador", + "currencyCode": "USD" + }, + { + "countryCode": "SX", + "countryName": "Sint Maarten", + "currencyCode": "ANG" + }, + { + "countryCode": "SY", + "countryName": "Syria", + "currencyCode": "SYP" + }, + { + "countryCode": "SZ", + "countryName": "Swaziland", + "currencyCode": "SZL" + }, + { + "countryCode": "TC", + "countryName": "Turks and Caicos Islands", + "currencyCode": "USD" + }, + { + "countryCode": "TD", + "countryName": "Chad", + "currencyCode": "XAF" + }, + { + "countryCode": "TF", + "countryName": "French Southern Territories", + "currencyCode": "EUR" + }, + { + "countryCode": "TG", + "countryName": "Togo", + "currencyCode": "XOF" + }, + { + "countryCode": "TH", + "countryName": "Thailand", + "currencyCode": "THB" + }, + { + "countryCode": "TJ", + "countryName": "Tajikistan", + "currencyCode": "TJS" + }, + { + "countryCode": "TK", + "countryName": "Tokelau", + "currencyCode": "NZD" + }, + { + "countryCode": "TL", + "countryName": "East Timor", + "currencyCode": "USD" + }, + { + "countryCode": "TM", + "countryName": "Turkmenistan", + "currencyCode": "TMT" + }, + { + "countryCode": "TN", + "countryName": "Tunisia", + "currencyCode": "TND" + }, + { + "countryCode": "TO", + "countryName": "Tonga", + "currencyCode": "TOP" + }, + { + "countryCode": "TR", + "countryName": "Turkey", + "currencyCode": "TRY" + }, + { + "countryCode": "TT", + "countryName": "Trinidad and Tobago", + "currencyCode": "TTD" + }, + { + "countryCode": "TV", + "countryName": "Tuvalu", + "currencyCode": "AUD" + }, + { + "countryCode": "TW", + "countryName": "Taiwan", + "currencyCode": "TWD" + }, + { + "countryCode": "TZ", + "countryName": "Tanzania", + "currencyCode": "TZS" + }, + { + "countryCode": "UA", + "countryName": "Ukraine", + "currencyCode": "UAH" + }, + { + "countryCode": "UG", + "countryName": "Uganda", + "currencyCode": "UGX" + }, + { + "countryCode": "UM", + "countryName": "U.S. Minor Outlying Islands", + "currencyCode": "USD" + }, + { + "countryCode": "US", + "countryName": "United States", + "currencyCode": "USD" + }, + { + "countryCode": "UY", + "countryName": "Uruguay", + "currencyCode": "UYU" + }, + { + "countryCode": "UZ", + "countryName": "Uzbekistan", + "currencyCode": "UZS" + }, + { + "countryCode": "VA", + "countryName": "Vatican City", + "currencyCode": "EUR" + }, + { + "countryCode": "VC", + "countryName": "Saint Vincent and the Grenadines", + "currencyCode": "XCD" + }, + { + "countryCode": "VE", + "countryName": "Venezuela", + "currencyCode": "VEF" + }, + { + "countryCode": "VG", + "countryName": "British Virgin Islands", + "currencyCode": "USD" + }, + { + "countryCode": "VI", + "countryName": "U.S. Virgin Islands", + "currencyCode": "USD" + }, + { + "countryCode": "VN", + "countryName": "Vietnam", + "currencyCode": "VND" + }, + { + "countryCode": "VU", + "countryName": "Vanuatu", + "currencyCode": "VUV" + }, + { + "countryCode": "WF", + "countryName": "Wallis and Futuna", + "currencyCode": "XPF" + }, + { + "countryCode": "WS", + "countryName": "Samoa", + "currencyCode": "WST" + }, + { + "countryCode": "XK", + "countryName": "Kosovo", + "currencyCode": "EUR" + }, + { + "countryCode": "YE", + "countryName": "Yemen", + "currencyCode": "YER" + }, + { + "countryCode": "YT", + "countryName": "Mayotte", + "currencyCode": "EUR" + }, + { + "countryCode": "ZA", + "countryName": "South Africa", + "currencyCode": "ZAR" + }, + { + "countryCode": "ZM", + "countryName": "Zambia", + "currencyCode": "ZMW" + }, + { + "countryCode": "ZW", + "countryName": "Zimbabwe", + "currencyCode": "ZWL" + } + ] +} \ No newline at end of file diff --git a/services/module-cms/functions/countries/country.components.ts b/services/module-cms/functions/countries/country.components.ts new file mode 100644 index 0000000..4594403 --- /dev/null +++ b/services/module-cms/functions/countries/country.components.ts @@ -0,0 +1,177 @@ +import { registerEnumType } from 'type-graphql'; + +export enum CurrencyCodesComponentEnum { + 'AED' = 'AED', + 'AFN' = 'AFN', + 'ALL' = 'ALL', + 'AMD' = 'AMD', + 'ANG' = 'ANG', + 'AOA' = 'AOA', + 'ARS' = 'ARS', + 'AUD' = 'AUD', + 'AWG' = 'AWG', + 'AZN' = 'AZN', + 'BAM' = 'BAM', + 'BBD' = 'BBD', + 'BDT' = 'BDT', + 'BGN' = 'BGN', + 'BHD' = 'BHD', + 'BIF' = 'BIF', + 'BMD' = 'BMD', + 'BND' = 'BND', + 'BOB' = 'BOB', + 'BRL' = 'BRL', + 'BSD' = 'BSD', + 'BTC' = 'BTC', + 'BTN' = 'BTN', + 'BWP' = 'BWP', + 'BYN' = 'BYN', + 'BZD' = 'BZD', + 'CAD' = 'CAD', + 'CDF' = 'CDF', + 'CHF' = 'CHF', + 'CLF' = 'CLF', + 'CLP' = 'CLP', + 'CNH' = 'CNH', + 'CNY' = 'CNY', + 'COP' = 'COP', + 'CRC' = 'CRC', + 'CUC' = 'CUC', + 'CUP' = 'CUP', + 'CVE' = 'CVE', + 'CZK' = 'CZK', + 'DJF' = 'DJF', + 'DKK' = 'DKK', + 'DOP' = 'DOP', + 'DZD' = 'DZD', + 'EGP' = 'EGP', + 'ERN' = 'ERN', + 'ETB' = 'ETB', + 'EUR' = 'EUR', + 'FJD' = 'FJD', + 'FKP' = 'FKP', + 'GBP' = 'GBP', + 'GEL' = 'GEL', + 'GGP' = 'GGP', + 'GHS' = 'GHS', + 'GIP' = 'GIP', + 'GMD' = 'GMD', + 'GNF' = 'GNF', + 'GTQ' = 'GTQ', + 'GYD' = 'GYD', + 'HKD' = 'HKD', + 'HNL' = 'HNL', + 'HRK' = 'HRK', + 'HTG' = 'HTG', + 'HUF' = 'HUF', + 'IDR' = 'IDR', + 'ILS' = 'ILS', + 'IMP' = 'IMP', + 'INR' = 'INR', + 'IQD' = 'IQD', + 'IRR' = 'IRR', + 'ISK' = 'ISK', + 'JEP' = 'JEP', + 'JMD' = 'JMD', + 'JOD' = 'JOD', + 'JPY' = 'JPY', + 'KES' = 'KES', + 'KGS' = 'KGS', + 'KHR' = 'KHR', + 'KMF' = 'KMF', + 'KPW' = 'KPW', + 'KRW' = 'KRW', + 'KWD' = 'KWD', + 'KYD' = 'KYD', + 'KZT' = 'KZT', + 'LAK' = 'LAK', + 'LBP' = 'LBP', + 'LKR' = 'LKR', + 'LRD' = 'LRD', + 'LSL' = 'LSL', + 'LYD' = 'LYD', + 'MAD' = 'MAD', + 'MDL' = 'MDL', + 'MGA' = 'MGA', + 'MKD' = 'MKD', + 'MMK' = 'MMK', + 'MNT' = 'MNT', + 'MOP' = 'MOP', + 'MRO' = 'MRO', + 'MRU' = 'MRU', + 'MUR' = 'MUR', + 'MVR' = 'MVR', + 'MWK' = 'MWK', + 'MXN' = 'MXN', + 'MYR' = 'MYR', + 'MZN' = 'MZN', + 'NAD' = 'NAD', + 'NGN' = 'NGN', + 'NIO' = 'NIO', + 'NOK' = 'NOK', + 'NPR' = 'NPR', + 'NZD' = 'NZD', + 'OMR' = 'OMR', + 'PAB' = 'PAB', + 'PEN' = 'PEN', + 'PGK' = 'PGK', + 'PHP' = 'PHP', + 'PKR' = 'PKR', + 'PLN' = 'PLN', + 'PYG' = 'PYG', + 'QAR' = 'QAR', + 'RON' = 'RON', + 'RSD' = 'RSD', + 'RUB' = 'RUB', + 'RWF' = 'RWF', + 'SAR' = 'SAR', + 'SBD' = 'SBD', + 'SCR' = 'SCR', + 'SDG' = 'SDG', + 'SEK' = 'SEK', + 'SGD' = 'SGD', + 'SHP' = 'SHP', + 'SLL' = 'SLL', + 'SOS' = 'SOS', + 'SRD' = 'SRD', + 'SSP' = 'SSP', + 'STD' = 'STD', + 'STN' = 'STN', + 'SVC' = 'SVC', + 'SYP' = 'SYP', + 'SZL' = 'SZL', + 'THB' = 'THB', + 'TJS' = 'TJS', + 'TMT' = 'TMT', + 'TND' = 'TND', + 'TOP' = 'TOP', + 'TRY' = 'TRY', + 'TTD' = 'TTD', + 'TWD' = 'TWD', + 'TZS' = 'TZS', + 'UAH' = 'UAH', + 'UGX' = 'UGX', + 'USD' = 'USD', + 'UYU' = 'UYU', + 'UZS' = 'UZS', + 'VEF' = 'VEF', + 'VND' = 'VND', + 'VUV' = 'VUV', + 'WST' = 'WST', + 'XAF' = 'XAF', + 'XAG' = 'XAG', + 'XAU' = 'XAU', + 'XCD' = 'XCD', + 'XDR' = 'XDR', + 'XOF' = 'XOF', + 'XPD' = 'XPD', + 'XPF' = 'XPF', + 'XPT' = 'XPT', + 'YER' = 'YER', + 'ZAR' = 'ZAR', + 'ZMW' = 'ZMW', + 'ZWL' = 'ZWL', +} +registerEnumType(CurrencyCodesComponentEnum, { + name: 'CurrencyCodesComponentEnum', +}); diff --git a/services/module-cms/functions/countries/country.model.ts b/services/module-cms/functions/countries/country.model.ts new file mode 100644 index 0000000..0299e5f --- /dev/null +++ b/services/module-cms/functions/countries/country.model.ts @@ -0,0 +1,41 @@ +import { ObjectType } from 'type-graphql'; + +import { Permission } from '@seed/interfaces/permission'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { EngineModel } from '@lib/seed/engine/EngineModel'; +import { plainToClass } from 'class-transformer'; +import { CustomSearchEngine } from '@seed/services/database/DBRequestService'; +import { IEngineSchema } from '@seed/engine/EngineSchema'; +import { ModelCollectionEnum } from '@src/__indexes/__collections'; +import { CountryDBInterfaceSchema, CountryDBSchema, CountrySchema } from './schemas/country.schema'; +import { CountryArgsSchema } from './schemas/country.input'; + +export const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.admin, AccountTypeEnum.public], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class CountryModel extends EngineModel { + public constructor(input?: CountryDBInterfaceSchema & Partial) { + const dataInit = plainToClass(CountryDBSchema, input || {}); + super({ + // ...init, + collectionName: 'countries', + permissions: permissions, + dataInit, + }); + } + + plainToClass(plain: any): CountrySchema | CountrySchema[] { + return plainToClass(CountrySchema, plain); + } + + searchEngine(): CustomSearchEngine { + const sEngine: CustomSearchEngine = {}; + + return sEngine; + } +} diff --git a/services/module-cms/functions/countries/country.service.ts b/services/module-cms/functions/countries/country.service.ts new file mode 100644 index 0000000..8108d19 --- /dev/null +++ b/services/module-cms/functions/countries/country.service.ts @@ -0,0 +1,28 @@ +import countries from './countries.json'; +import { CountrySchema } from './schemas/country.schema'; +import { v4 as uuid } from 'uuid'; +import { permissions } from './country.model'; +import { ApolloContext } from '@seed/interfaces/context'; + +const formatData = (data: any, ctx): CountrySchema => { + const account = ctx.ctx.user; + const formatedData: any = { ...data, _id: uuid(), createdAt: new Date(), by: account._id, updatedAt: new Date(), permissions }; + return formatedData; +}; + +export const importFromJSON = (ctx: ApolloContext): any => { + const formatedCountries = countries.countries.map((country) => { + return formatData( + { + currencyCodes: [country.currencyCode], + countryCode: country.countryCode, + countryName: { + en: country.countryName, + }, + }, + ctx, + ); + }); + + return formatedCountries; +}; diff --git a/services/module-cms/functions/countries/resolvers/country.resolver.admin.ts b/services/module-cms/functions/countries/resolvers/country.resolver.admin.ts new file mode 100644 index 0000000..1170fc6 --- /dev/null +++ b/services/module-cms/functions/countries/resolvers/country.resolver.admin.ts @@ -0,0 +1,73 @@ +import { Arg, Authorized, Ctx, Mutation, Query, Resolver } from 'type-graphql'; + +import CountryModel from '../country.model'; +import { createEngineQueryResolver, createEngineMutationResolver } from '@seed/engine/genericResolvers/BaseEngineResolver'; +import { CountryArgsSchema, CountryEditInputSchema, CountryNewInputSchema } from '../schemas/country.input'; +import { CountrySchema } from '../schemas/country.schema'; +import { ApolloContext } from '@seed/interfaces/context'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { importFromJSON } from '../country.service'; +import { SimpleResult } from '@seed/interfaces/components'; + +const CountryGenericQueryResolver = createEngineQueryResolver({ + domain: 'country', + schemaName: CountrySchema, + modelName: CountryModel, + argsType: CountryArgsSchema, + engineMiddleware: { + noPermissionCheck: false, + authorization: [AccountTypeEnum.admin], + }, +}); + +const CountryGenericMutationResolver = createEngineMutationResolver({ + domain: 'country', + schemaName: CountrySchema, + modelName: CountryModel, + editInput: CountryEditInputSchema, + newInput: CountryNewInputSchema, + engineMiddleware: { + noPermissionCheck: false, + authorization: [AccountTypeEnum.admin], + }, +}); + +@Resolver(CountrySchema) +export class CountryQueryResolver extends CountryGenericQueryResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ +} + +@Resolver(CountrySchema) +export class CountryMutationResolver extends CountryGenericMutationResolver { + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + + @Mutation(() => SimpleResult) + @Authorized(AccountTypeEnum.admin) + async countryImportMany(@Ctx() ctx: ApolloContext): Promise { + const dataToImport: CountrySchema[] = importFromJSON(ctx); + const model = new CountryModel(); + + try { + await (await model.db()).insertMany(dataToImport, { ordered: false }); + return { message: 'imported' }; + } catch (error) { + throw error; + } + } +} + +export const CountryAdminResolvers = [CountryQueryResolver, CountryMutationResolver]; diff --git a/services/module-cms/functions/countries/resolvers/country.resolver.ts b/services/module-cms/functions/countries/resolvers/country.resolver.ts new file mode 100644 index 0000000..f3ea3a3 --- /dev/null +++ b/services/module-cms/functions/countries/resolvers/country.resolver.ts @@ -0,0 +1,44 @@ +import { Resolver } from 'type-graphql'; + +import CountryModel from '../country.model'; +import { createEngineQueryResolver } from '@seed/engine/genericResolvers/BaseEngineResolver'; +import { CountryArgsSchema } from '../schemas/country.input'; +import { CountrySchema } from '../schemas/country.schema'; +import { AccountTypeEnum } from '@src/accounts/account.components'; + +const CountryGenericQueryResolver = createEngineQueryResolver({ + domain: 'countries', + schemaName: CountrySchema, + modelName: CountryModel, + argsType: CountryArgsSchema, + engineMiddleware: { + noPermissionCheck: false, + authorization: [AccountTypeEnum.public], + }, +}); + +@Resolver(CountrySchema) +export class CountryQueryResolver extends CountryGenericQueryResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ +} + +@Resolver(CountrySchema) +export class CountryMutationResolver { + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ +} + +export const CountryAppResolvers = [CountryQueryResolver]; diff --git a/services/module-cms/functions/countries/schemas/country.input.ts b/services/module-cms/functions/countries/schemas/country.input.ts new file mode 100644 index 0000000..caba921 --- /dev/null +++ b/services/module-cms/functions/countries/schemas/country.input.ts @@ -0,0 +1,34 @@ +import { InputType, ArgsType, Field } from 'type-graphql'; + +import { GetManyArgs } from '@seed/graphql/Request'; +import { CountryBaseSchema } from './country.schema'; +import { CountryCodesComponentEnum } from '@seed/services/geolocation/components/countries'; +import { TranslatableComponent } from '@src/__components/components'; +import { CurrencyCodesComponentEnum } from '../country.components'; + +@InputType() +export class CountryNewInputSchema extends CountryBaseSchema { + @Field(() => TranslatableComponent) + countryName: TranslatableComponent; + + @Field(() => CountryCodesComponentEnum) + countryCode: CountryCodesComponentEnum; + + @Field(() => [CurrencyCodesComponentEnum]) + currencyCodes: CurrencyCodesComponentEnum[]; +} + +@InputType() +export class CountryEditInputSchema implements Partial { + @Field(() => TranslatableComponent, { nullable: true }) + countryName?: TranslatableComponent; + + @Field(() => CountryCodesComponentEnum, { nullable: true }) + countryCode?: CountryCodesComponentEnum; + + @Field(() => [CurrencyCodesComponentEnum], { nullable: true }) + currencyCodes?: CurrencyCodesComponentEnum[]; +} + +@ArgsType() +export class CountryArgsSchema extends GetManyArgs {} diff --git a/services/module-cms/functions/countries/schemas/country.schema.ts b/services/module-cms/functions/countries/schemas/country.schema.ts new file mode 100644 index 0000000..6be84b0 --- /dev/null +++ b/services/module-cms/functions/countries/schemas/country.schema.ts @@ -0,0 +1,34 @@ +import { IEngineSchema, MetaBy, MetaPermissions } from '@seed/engine/EngineSchema'; +import { Field, InputType, ObjectType } from 'type-graphql'; +import { CountryCodesComponentEnum } from '@seed/services/geolocation/components/countries'; +import { TranslatableComponent } from '@src/__components/components'; +import { CurrencyCodesComponentEnum } from '../country.components'; + +@InputType() +@ObjectType() +export class CountryBaseSchema { + @Field(() => TranslatableComponent) + countryName: TranslatableComponent; + + @Field(() => CountryCodesComponentEnum) + countryCode: CountryCodesComponentEnum; + + @Field(() => [CurrencyCodesComponentEnum]) + currencyCodes: CurrencyCodesComponentEnum[]; +} + +@ObjectType() +export class CountryDBInterfaceSchema extends CountryBaseSchema {} + +@ObjectType() +export class CountryDBSchema extends CountryDBInterfaceSchema {} + +@ObjectType({ implements: IEngineSchema }) +export class CountrySchema extends CountryDBSchema implements IEngineSchema { + _id: string; + organisationId?: string | undefined; + by?: MetaBy | undefined; + permissions: MetaPermissions; + createdAt: Date; + updatedAt: Date; +} diff --git a/services/module-cms/functions/emails-settings/emails-settings.components.ts b/services/module-cms/functions/emails-settings/emails-settings.components.ts new file mode 100644 index 0000000..bb1f26f --- /dev/null +++ b/services/module-cms/functions/emails-settings/emails-settings.components.ts @@ -0,0 +1,57 @@ +import { AvailableTranslation, TranslatableComponent } from '@src/__components/components'; +import { Field, InputType, ObjectType, registerEnumType } from 'type-graphql'; + +@ObjectType() +@InputType('PushNotifComponentInput') +export class PushNotifComponent { + @Field(() => TranslatableComponent, { nullable: true }) + subject?: TranslatableComponent; + + @Field(() => TranslatableComponent) + text: TranslatableComponent; +} + +@ObjectType() +@InputType('SmsSettingsComponentInput') +export class SmsSettingsComponent { + @Field(() => TranslatableComponent) + text: TranslatableComponent; +} + +export enum SlackActionStyleEnum { + primary = 'primary', + danger = 'danger', +} +registerEnumType(SlackActionStyleEnum, { + name: 'SlackActionStyleEnum', +}); + +@ObjectType() +@InputType('SlackActionInfoInput') +export class SlackActionInfo { + @Field() + text: string; + @Field() + url: string; + @Field(() => SlackActionStyleEnum, { nullable: true }) + style?: SlackActionStyleEnum; +} + +@ObjectType() +@InputType('SlackSettingsComponentInput') +export class SlackSettingsComponent { + @Field(() => [String]) + webhookUrls: string[]; + + @Field({ nullable: true }) + username?: string; + + @Field(() => String) + text: string; + + @Field({ nullable: true }) + icon?: string; + + @Field(() => [SlackActionInfo], { nullable: true }) + actions?: SlackActionInfo[]; +} diff --git a/services/module-cms/functions/emails-settings/emails-settings.model.ts b/services/module-cms/functions/emails-settings/emails-settings.model.ts new file mode 100644 index 0000000..4f123ba --- /dev/null +++ b/services/module-cms/functions/emails-settings/emails-settings.model.ts @@ -0,0 +1,191 @@ +import { ObjectType, Field, ID, ArgsType, InputType } from 'type-graphql'; + +import { BaseGraphModel } from '@seed/graphql/BaseGraphModel'; +import { Permission } from '@seed/interfaces/permission'; + +import { GetArgs, GetManyArgs } from '@seed/graphql/Request'; + +import { SettingsType } from '@seed/interfaces/components'; +import { RessourceEnum, TranslatableComponent } from '@src/__components/components'; +import { ApolloError } from 'apollo-server-lambda'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import _ from 'lodash'; +import { PushNotifComponent, SlackSettingsComponent, SmsSettingsComponent } from './emails-settings.components'; +import { newError } from '@seed/helpers/Error'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.admin], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class EmailSettingsModel extends BaseGraphModel { + public constructor() { + super({ + // ...init, + collectionName: 'settings', + permissions: permissions, + }); + } + // eslint-disable-next-line @typescript-eslint/no-unused-vars + @Field(() => ID) + readonly _id: string; + + @Field(() => SettingsType) + type = SettingsType.emails; + + @Field() + key: string; + + @Field(() => TranslatableComponent, { nullable: true }) + subject?: TranslatableComponent; + @Field(() => TranslatableComponent, { nullable: true }) + body?: TranslatableComponent; + + // Related to EMAIL + @Field() + custom: boolean; + @Field(() => TranslatableComponent, { nullable: true }) + templateId?: TranslatableComponent; + @Field() + fromEmail: string; + @Field() + fromName: string; + @Field() + replyToEmail: string; + @Field(() => [String], { nullable: true }) + cci: string[]; + + @Field(() => [String], { nullable: true }) + availableFields?: string[]; + + @Field(() => String, { nullable: true }) + templateSample?: string; + + /* + SMS + */ + + @Field(() => SmsSettingsComponent, { nullable: true }) + smsSettings?: SmsSettingsComponent; + + /* + PUSH NOTIFICATION + */ + + @Field(() => PushNotifComponent, { nullable: true }) + pushSettings?: PushNotifComponent; + + /* + SLACK + */ + @Field(() => SlackSettingsComponent, { nullable: true }) + slackSettings?: SlackSettingsComponent; + + searchOptions(): string[] { + return ['key', 'type']; + } + filterOptions(): string[] { + return ['key', 'type']; + } + + onCreate(input: NewEmailSettingsInput): void { + if (input.custom == false) { + if (!input.subject || !input.body) throw newError(3000, { message: 'subject and body must be required when not custom' }); + } + return; + } +} + +@InputType() +export class NewEmailSettingsInput { + @Field(() => TranslatableComponent, { nullable: true }) + subject: TranslatableComponent; + @Field(() => TranslatableComponent, { nullable: true }) + body: TranslatableComponent; + + @Field(() => String) + type = SettingsType.emails; + + @Field() + key: string; + + @Field({ nullable: true }) + custom: boolean; + + @Field(() => TranslatableComponent, { nullable: true }) + templateId?: TranslatableComponent; + @Field() + fromEmail: string; + @Field() + fromName: string; + @Field() + replyToEmail: string; + @Field(() => [String], { nullable: true }) + cci: string[]; + + /* + SMS + */ + + @Field(() => SmsSettingsComponent, { nullable: true }) + smsSettings?: SmsSettingsComponent; + + /* + SLACK + */ + @Field(() => SlackSettingsComponent, { nullable: true }) + slackSettings?: SlackSettingsComponent; +} + +@InputType() +export class EditEmailSettingsInput { + @Field(() => TranslatableComponent, { nullable: true }) + subject: TranslatableComponent; + @Field(() => TranslatableComponent, { nullable: true }) + body: TranslatableComponent; + + @Field(() => String) + type = SettingsType.emails; + + @Field({ nullable: true }) + custom: boolean; + + @Field(() => TranslatableComponent, { nullable: true }) + templateId?: TranslatableComponent; + + @Field() + fromEmail: string; + @Field() + fromName: string; + @Field() + replyToEmail: string; + @Field(() => [String], { nullable: true }) + cci: string[]; + + /* + SMS + */ + + @Field(() => SmsSettingsComponent, { nullable: true }) + smsSettings?: SmsSettingsComponent; + + /* + SLACK + */ + @Field(() => SlackSettingsComponent, { nullable: true }) + slackSettings?: SlackSettingsComponent; +} + +@ArgsType() +export class EmailSettingsArgs { + @Field({ nullable: true }) + search?: string; + + type = SettingsType.emails; + + @Field(() => GetArgs, { nullable: true }) + pagination?: GetArgs; +} diff --git a/services/module-cms/functions/emails-settings/emails-settings.resolver.ts b/services/module-cms/functions/emails-settings/emails-settings.resolver.ts new file mode 100644 index 0000000..8d5c279 --- /dev/null +++ b/services/module-cms/functions/emails-settings/emails-settings.resolver.ts @@ -0,0 +1,33 @@ +import { Resolver, Query, Args, Authorized, FieldResolver, Root, Ctx } from 'type-graphql'; +import EmailSettingsModel, { EditEmailSettingsInput, EmailSettingsArgs, NewEmailSettingsInput } from './emails-settings.model'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { createBaseResolver } from '@seed/graphql/baseResolvers/BaseResolver'; + +const EmailSettingBaseResolver = createBaseResolver( + 'emailsSettings', + EmailSettingsModel, + EmailSettingsArgs, + NewEmailSettingsInput, + EditEmailSettingsInput, + [AccountTypeEnum.admin], +); + +@Resolver(EmailSettingsModel) +export default class EmailSettingsAdminResolver extends EmailSettingBaseResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ +} diff --git a/services/module-cms/functions/error-settings/errors-settings.components.ts b/services/module-cms/functions/error-settings/errors-settings.components.ts new file mode 100644 index 0000000..e69de29 diff --git a/services/module-cms/functions/error-settings/errors-settings.model.ts b/services/module-cms/functions/error-settings/errors-settings.model.ts new file mode 100644 index 0000000..101de69 --- /dev/null +++ b/services/module-cms/functions/error-settings/errors-settings.model.ts @@ -0,0 +1,91 @@ +import { ObjectType, Field, ID, ArgsType, InputType } from 'type-graphql'; + +import { BaseGraphModel } from '@seed/graphql/BaseGraphModel'; +import { Permission } from '@seed/interfaces/permission'; + +import { GetArgs } from '@seed/graphql/Request'; + +import { SettingsType } from '@seed/interfaces/components'; +import { TranslatableComponent } from '@src/__components/components'; +import { AccountTypeEnum } from '@src/accounts/account.components'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.admin], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class ErrorSettingsModel extends BaseGraphModel { + public constructor() { + super({ + // ...init, + collectionName: 'settings', + permissions: permissions, + }); + } + // eslint-disable-next-line @typescript-eslint/no-unused-vars + @Field(() => ID) + readonly _id: string; + + @Field(() => SettingsType) + type = SettingsType.errors; + + @Field() + key: string; + + @Field() + value: string; + + @Field(() => TranslatableComponent, { nullable: true }) + subject?: TranslatableComponent; + @Field(() => TranslatableComponent, { nullable: true }) + body: TranslatableComponent; + + @Field(() => [String], { nullable: true }) + availableFields?: string[]; + + searchOptions(): string[] { + return ['key', 'type']; + } + filterOptions(): string[] { + return ['key', 'type']; + } +} + +@InputType() +export class NewErrorSettingsInput { + @Field(() => TranslatableComponent, { nullable: true }) + subject?: TranslatableComponent; + @Field(() => TranslatableComponent) + body: TranslatableComponent; + + @Field(() => String) + type = SettingsType.errors; + + @Field() + key: string; + + @Field() + value: string; +} + +@InputType() +export class EditErrorSettingsInput { + @Field(() => TranslatableComponent, { nullable: true }) + subject: TranslatableComponent; + @Field(() => TranslatableComponent, { nullable: true }) + body: TranslatableComponent; +} + +@ArgsType() +export class ErrorSettingsArgs { + @Field({ nullable: true }) + search?: string; + + type = SettingsType.errors; + + @Field(() => GetArgs, { nullable: true }) + pagination?: GetArgs; +} diff --git a/services/module-cms/functions/error-settings/errors-settings.resolver.ts b/services/module-cms/functions/error-settings/errors-settings.resolver.ts new file mode 100644 index 0000000..97445e7 --- /dev/null +++ b/services/module-cms/functions/error-settings/errors-settings.resolver.ts @@ -0,0 +1,33 @@ +import { Resolver } from 'type-graphql'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { createBaseResolver } from '@seed/graphql/baseResolvers/BaseResolver'; +import ErrorSettingsModel, { EditErrorSettingsInput, ErrorSettingsArgs, NewErrorSettingsInput } from './errors-settings.model'; + +const ErrorSettingsBaseResolver = createBaseResolver( + 'errorSettings', + ErrorSettingsModel, + ErrorSettingsArgs, + NewErrorSettingsInput, + EditErrorSettingsInput, + [AccountTypeEnum.admin], +); + +@Resolver(ErrorSettingsModel) +export default class ErrorSettingsAdminResolver extends ErrorSettingsBaseResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ +} diff --git a/services/module-cms/functions/lists/list.model.ts b/services/module-cms/functions/lists/list.model.ts new file mode 100644 index 0000000..0e5df1a --- /dev/null +++ b/services/module-cms/functions/lists/list.model.ts @@ -0,0 +1,75 @@ +import { ObjectType, Field, ID, ArgsType, InputType, Int } from 'type-graphql'; + +import { GetArgs } from '@seed/graphql/Request'; + +import { TranslatableComponent, ListEnum } from '@src/__components/components'; + +import { Permission } from '@seed/interfaces/permission'; +import BaseSEOModel, { EditBaseSEOInput, NewBaseSEOInput } from '@seed/graphql/baseModels/BaseSeoModel'; +import { AccountTypeEnum } from '@src/accounts/account.components'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.public], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class ListModel extends BaseSEOModel { + public constructor() { + super('lists', permissions); + } + + @Field(() => ListEnum) + ressourceType: ListEnum; + + @Field(() => TranslatableComponent) + title: TranslatableComponent; + + @Field({ nullable: true }) + colorCode?: string; + + searchOptions(): string[] { + return ['ressourceType']; + } + filterOptions(): string[] { + return ['ressourceType']; + } +} + +@ArgsType() +export class ListArgs { + @Field({ nullable: true }) + search?: string; + + @Field(() => ListEnum) + ressourceType: ListEnum; + + @Field(() => GetArgs, { nullable: true }) + pagination?: GetArgs; +} + +@InputType() +export class NewListInput extends NewBaseSEOInput implements Partial { + @Field(() => ListEnum) + ressourceType: ListEnum; + + @Field(() => TranslatableComponent) + title: TranslatableComponent; + + @Field({ nullable: true }) + colorCode?: string; +} + +@InputType() +export class EditListInput extends EditBaseSEOInput implements Partial { + @Field(() => ListEnum, { nullable: true }) + ressourceType?: ListEnum; + + @Field(() => TranslatableComponent, { nullable: true }) + title?: TranslatableComponent; + + @Field({ nullable: true }) + colorCode?: string; +} diff --git a/services/module-cms/functions/lists/list.resolver.admin.ts b/services/module-cms/functions/lists/list.resolver.admin.ts new file mode 100644 index 0000000..26c6cab --- /dev/null +++ b/services/module-cms/functions/lists/list.resolver.admin.ts @@ -0,0 +1,26 @@ +import { Resolver } from 'type-graphql'; +import { createBaseResolver } from '@seed/graphql/baseResolvers/BaseResolver'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import ListModel, { EditListInput, ListArgs, NewListInput } from './list.model'; + +const ListAdminBaseResolver = createBaseResolver('lists', ListModel, ListArgs, NewListInput, EditListInput, [AccountTypeEnum.admin]); + +@Resolver(ListModel) +export default class ListAdminResolver extends ListAdminBaseResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ +} diff --git a/services/module-cms/functions/lists/list.resolver.fields.ts b/services/module-cms/functions/lists/list.resolver.fields.ts new file mode 100644 index 0000000..8e3e3d2 --- /dev/null +++ b/services/module-cms/functions/lists/list.resolver.fields.ts @@ -0,0 +1,15 @@ +import { Resolver } from 'type-graphql'; + +import ListModel from './list.model'; + +@Resolver(ListModel) +export default class ListFieldResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ +} diff --git a/services/module-cms/functions/lists/list.resolver.ts b/services/module-cms/functions/lists/list.resolver.ts new file mode 100644 index 0000000..4ec5a99 --- /dev/null +++ b/services/module-cms/functions/lists/list.resolver.ts @@ -0,0 +1,26 @@ +import { Resolver } from 'type-graphql'; + +import { createBasePublicResolver } from '@seed/graphql/baseResolvers/BasePublicResolver'; +import ListModel, { ListArgs } from './list.model'; + +const ListBaseResolver = createBasePublicResolver('lists', ListModel, ListArgs); + +@Resolver(ListModel) +export default class ListResolver extends ListBaseResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ +} diff --git a/services/module-cms/functions/pages/pages.model.ts b/services/module-cms/functions/pages/pages.model.ts new file mode 100644 index 0000000..5900639 --- /dev/null +++ b/services/module-cms/functions/pages/pages.model.ts @@ -0,0 +1,51 @@ +import { ObjectType, Field, ID, ArgsType, InputType, registerEnumType, Int } from 'type-graphql'; + +import { BaseGraphModel } from '@seed/graphql/BaseGraphModel'; +import { Permission } from '@seed/interfaces/permission'; + +import { GetArgs } from '@seed/graphql/Request'; + +import { GraphQLJSONObject } from 'graphql-type-json'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { EngineModel } from '@seed/engine/EngineModel'; +import { PageDBInterfaceSchema, PageDBSchema, PageSchema } from './schemas/pages.schema'; +import { plainToClass } from 'class-transformer'; +import { IEngineSchema } from '@seed/engine/EngineSchema'; +import { uploadToS3 } from '@services/api-uploads/services/S3Service'; +import { updateCaching } from '@services/module-cms/services/PageService'; +import { StreamDBSchema } from '@seed/engine/utils/streams/schemas/stream.schema'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.public], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class PageModel extends EngineModel { + public constructor(input?: PageDBInterfaceSchema & Partial) { + const dataInit = plainToClass(PageDBSchema, input || {}); + super({ + collectionName: 'pages', + permissions: permissions, + dataInit, + }); + } + + plainToClass(plain: any): PageSchema | PageSchema[] { + return plainToClass(PageSchema, plain); + } + + async afterCreate(): Promise { + await updateCaching(this); + } + + async afterUpdate(): Promise { + await updateCaching(this); + } + + async afterDelete(): Promise { + await updateCaching(this); + } +} diff --git a/services/module-cms/functions/pages/pages.resolver.admin.ts b/services/module-cms/functions/pages/pages.resolver.admin.ts new file mode 100644 index 0000000..7680b2d --- /dev/null +++ b/services/module-cms/functions/pages/pages.resolver.admin.ts @@ -0,0 +1,72 @@ +import { Resolver, Query, Args, Authorized, Ctx } from 'type-graphql'; +import { Mutation, Arg } from 'type-graphql'; + +import { getOneGeneric, getManyGenericWithArgs, addOneGeneric, editOneGeneric, deleteOneGeneric } from '@seed/graphql/BaseService'; + +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { createEngineMutationResolver, createEngineQueryResolver } from '@seed/engine/genericResolvers/BaseEngineResolver'; +import { PageSchema } from './schemas/pages.schema'; +import { PageArgs, NewPageInput, EditPageInput } from './schemas/pages.input'; +import PageModel from './pages.model'; +import { SuccessResponse } from '@seed/interfaces/response'; +import { EngineMiddleware } from '@seed/graphql/MiddlewareV2'; +import { updateCaching as updatePagesCaching } from '@services/module-cms/services/PageService'; +import { ApolloContext } from '@seed/interfaces/context'; + +const PageBaseQueryAdminResolver = createEngineQueryResolver({ + domain: 'pages', + schemaName: PageSchema, + modelName: PageModel, + argsType: PageArgs, + engineMiddleware: { + authorization: [AccountTypeEnum.admin], + }, +}); + +const PageBaseMutationAdminResolver = createEngineMutationResolver({ + domain: 'pages', + schemaName: PageSchema, + modelName: PageModel, + newInput: NewPageInput, + editInput: EditPageInput, + engineMiddleware: { + authorization: [AccountTypeEnum.admin], + }, +}); + +@Resolver(PageModel) +export class PageQueryAdminResolver extends PageBaseQueryAdminResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ +} + +@Resolver(PageModel) +export class PageMutationAdminResolver extends PageBaseMutationAdminResolver { + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + + @Mutation(() => SuccessResponse) + @EngineMiddleware({ + authorization: [AccountTypeEnum.admin], + }) + async refreshCaching(@Ctx() ctx: ApolloContext): Promise { + const model = new PageModel(); + await updatePagesCaching(model); + + return { success: true }; + } +} + +export const PageAdminResolvers = [PageQueryAdminResolver, PageMutationAdminResolver]; diff --git a/services/module-cms/functions/pages/schemas/pages.input.ts b/services/module-cms/functions/pages/schemas/pages.input.ts new file mode 100644 index 0000000..aa6414b --- /dev/null +++ b/services/module-cms/functions/pages/schemas/pages.input.ts @@ -0,0 +1,16 @@ +import { InputType, ArgsType, Field } from 'type-graphql'; + +import { GetManyArgs } from '@seed/graphql/Request'; +import { PageBaseSchema } from './pages.schema'; + +@InputType() +export class NewPageInput extends PageBaseSchema {} + +@InputType() +export class EditPageInput extends PageBaseSchema { + @Field({ nullable: true }) + title: string; +} + +@ArgsType() +export class PageArgs extends GetManyArgs {} diff --git a/services/module-cms/functions/pages/schemas/pages.schema.ts b/services/module-cms/functions/pages/schemas/pages.schema.ts new file mode 100644 index 0000000..7f33ec3 --- /dev/null +++ b/services/module-cms/functions/pages/schemas/pages.schema.ts @@ -0,0 +1,38 @@ +import { IEngineSchema, MetaBy, MetaPermissions } from '@seed/engine/EngineSchema'; +import { GraphQLJSONObject } from 'graphql-type-json'; +import { InputType, ObjectType, Field, Int } from 'type-graphql'; + +@InputType() +@ObjectType() +export class PageBaseSchema { + @Field() + title: string; + + @Field({ nullable: true }) + label?: string; + + @Field(() => GraphQLJSONObject, { nullable: true }) + content?: any; + + @Field(() => Int, { nullable: true }) + position?: number; + + @Field({ nullable: true }) + toComplete?: boolean; +} + +@ObjectType() +export class PageDBInterfaceSchema extends PageBaseSchema {} + +@ObjectType() +export class PageDBSchema extends PageDBInterfaceSchema {} + +@ObjectType({ implements: IEngineSchema }) +export class PageSchema extends PageDBSchema implements IEngineSchema { + _id: string; + organisationId?: string | undefined; + by?: MetaBy | undefined; + permissions: MetaPermissions; + createdAt: Date; + updatedAt: Date; +} diff --git a/services/module-cms/functions/project-settings/project-settings.components.ts b/services/module-cms/functions/project-settings/project-settings.components.ts new file mode 100644 index 0000000..e69de29 diff --git a/services/module-cms/functions/project-settings/project-settings.model.ts b/services/module-cms/functions/project-settings/project-settings.model.ts new file mode 100644 index 0000000..2dc0da6 --- /dev/null +++ b/services/module-cms/functions/project-settings/project-settings.model.ts @@ -0,0 +1,71 @@ +import { ObjectType, Field, ID, ArgsType, InputType } from 'type-graphql'; + +import { BaseGraphModel } from '@seed/graphql/BaseGraphModel'; +import { Permission } from '@seed/interfaces/permission'; + +import { GetManyArgs } from '@seed/graphql/Request'; + +import { SettingsType } from '@seed/interfaces/components'; +import { AccountTypeEnum } from '@src/accounts/account.components'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.admin], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class ProjectSettingModel extends BaseGraphModel { + public constructor() { + super({ + // ...init, + collectionName: 'settings', + permissions: permissions, + }); + } + // eslint-disable-next-line @typescript-eslint/no-unused-vars + @Field(() => ID) + readonly _id: string; + + @Field(() => SettingsType) + type = SettingsType.env; + + @Field() + key: string; + + @Field() + value: string; + + searchOptions(): string[] { + return ['key', 'type']; + } + filterOptions(): string[] { + return ['key', 'type']; + } + + onCreate(): void { + return; + } +} + +@InputType() +export class NewProjectSettingsInput { + @Field() + key: string; + + @Field() + value: string; +} + +@InputType() +export class EditProjectSettingsInput { + @Field({ nullable: true }) + value: string; +} + +@ArgsType() +export class ProjectSettingsArgs extends GetManyArgs { + @Field(() => SettingsType) + type = SettingsType.env; +} diff --git a/services/module-cms/functions/project-settings/project-settings.resolver.admin.ts b/services/module-cms/functions/project-settings/project-settings.resolver.admin.ts new file mode 100644 index 0000000..fab13d0 --- /dev/null +++ b/services/module-cms/functions/project-settings/project-settings.resolver.admin.ts @@ -0,0 +1,35 @@ +import { Resolver, Query, Args, Authorized, FieldResolver, Root, Ctx } from 'type-graphql'; +import ProjectSettingssModel, { EditProjectSettingsInput, NewProjectSettingsInput, ProjectSettingsArgs } from './project-settings.model'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { GetManyArgs } from '@seed/graphql/Request'; +import { createBaseResolver } from '@seed/graphql/baseResolvers/BaseResolver'; +import ProjectSettingModel from './project-settings.model'; + +const ProjectSettingsBaseResolver = createBaseResolver( + 'projectSettings', + ProjectSettingModel, + ProjectSettingsArgs, + NewProjectSettingsInput, + EditProjectSettingsInput, + [AccountTypeEnum.admin], +); + +@Resolver(ProjectSettingssModel) +export default class ProjectSettingsAdminResolver extends ProjectSettingsBaseResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ +} diff --git a/services/module-cms/import/pages.countries.json b/services/module-cms/import/pages.countries.json new file mode 100644 index 0000000..d8e6c77 --- /dev/null +++ b/services/module-cms/import/pages.countries.json @@ -0,0 +1,12690 @@ +{ + "_id" : "011ddfb4-1c00-4123-8d8e-de446c41424d", + "content" : { + "_NEW" : { + "type" : "Array", + "position" : NumberInt(1), + "key" : "countries2", + "locked" : false, + "lockedValue" : false, + "open" : false, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "countries" : { + "label" : "countries", + "position" : NumberInt(2), + "type" : "Array", + "index" : NumberInt(1), + "content" : [ + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Afghanistan", + "fr" : "Afghanistan", + "nl" : "Afghanistan" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "AF" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Åland Islands", + "fr" : "Åland Islands", + "nl" : "Åland Islands" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "AX" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Albania", + "fr" : "Albania", + "nl" : "Albania" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "AL" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Algeria", + "fr" : "Algeria", + "nl" : "Algeria" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "DZ" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "American Samoa", + "fr" : "American Samoa", + "nl" : "American Samoa" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "AS" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Andorra", + "fr" : "Andorra", + "nl" : "Andorra" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "AD" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Angola", + "fr" : "Angola", + "nl" : "Angola" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "AO" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Anguilla", + "fr" : "Anguilla", + "nl" : "Anguilla" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "AI" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Antarctica", + "fr" : "Antarctica", + "nl" : "Antarctica" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "AQ" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Antigua and Barbuda", + "fr" : "Antigua and Barbuda", + "nl" : "Antigua and Barbuda" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "AG" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Argentina", + "fr" : "Argentina", + "nl" : "Argentina" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "AR" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Armenia", + "fr" : "Armenia", + "nl" : "Armenia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "AM" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Aruba", + "fr" : "Aruba", + "nl" : "Aruba" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "AW" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Australia", + "fr" : "Australia", + "nl" : "Australia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "AU" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Austria", + "fr" : "Austria", + "nl" : "Austria" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "AT" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Azerbaijan", + "fr" : "Azerbaijan", + "nl" : "Azerbaijan" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "AZ" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Bahamas", + "fr" : "Bahamas", + "nl" : "Bahamas" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "BS" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Bahrain", + "fr" : "Bahrain", + "nl" : "Bahrain" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "BH" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Bangladesh", + "fr" : "Bangladesh", + "nl" : "Bangladesh" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "BD" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Barbados", + "fr" : "Barbados", + "nl" : "Barbados" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "BB" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Belarus", + "fr" : "Belarus", + "nl" : "Belarus" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "BY" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Belgium", + "fr" : "Belgium", + "nl" : "Belgium" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "BE" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Belize", + "fr" : "Belize", + "nl" : "Belize" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "BZ" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Benin", + "fr" : "Benin", + "nl" : "Benin" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "BJ" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Bermuda", + "fr" : "Bermuda", + "nl" : "Bermuda" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "BM" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Bhutan", + "fr" : "Bhutan", + "nl" : "Bhutan" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "BT" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Bolivia", + "fr" : "Bolivia", + "nl" : "Bolivia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "BO" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Bosnia and Herzegovina", + "fr" : "Bosnia and Herzegovina", + "nl" : "Bosnia and Herzegovina" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "BA" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Botswana", + "fr" : "Botswana", + "nl" : "Botswana" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "BW" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Bouvet Island", + "fr" : "Bouvet Island", + "nl" : "Bouvet Island" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "BV" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Brazil", + "fr" : "Brazil", + "nl" : "Brazil" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "BR" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "British Indian Ocean Territory", + "fr" : "British Indian Ocean Territory", + "nl" : "British Indian Ocean Territory" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "IO" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Brunei Darussalam", + "fr" : "Brunei Darussalam", + "nl" : "Brunei Darussalam" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "BN" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Bulgaria", + "fr" : "Bulgaria", + "nl" : "Bulgaria" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "BG" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Burkina Faso", + "fr" : "Burkina Faso", + "nl" : "Burkina Faso" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "BF" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Burundi", + "fr" : "Burundi", + "nl" : "Burundi" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "BI" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Cambodia", + "fr" : "Cambodia", + "nl" : "Cambodia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "KH" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Cameroon", + "fr" : "Cameroon", + "nl" : "Cameroon" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "CM" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Canada", + "fr" : "Canada", + "nl" : "Canada" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "CA" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Cape Verde", + "fr" : "Cape Verde", + "nl" : "Cape Verde" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "CV" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Cayman Islands", + "fr" : "Cayman Islands", + "nl" : "Cayman Islands" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "KY" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Central African Republic", + "fr" : "Central African Republic", + "nl" : "Central African Republic" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "CF" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Chad", + "fr" : "Chad", + "nl" : "Chad" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "TD" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Chile", + "fr" : "Chile", + "nl" : "Chile" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "CL" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "China", + "fr" : "China", + "nl" : "China" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "CN" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Christmas Island", + "fr" : "Christmas Island", + "nl" : "Christmas Island" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "CX" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Cocos (Keeling) Islands", + "fr" : "Cocos (Keeling) Islands", + "nl" : "Cocos (Keeling) Islands" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "CC" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Colombia", + "fr" : "Colombia", + "nl" : "Colombia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "CO" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Comoros", + "fr" : "Comoros", + "nl" : "Comoros" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "KM" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Congo", + "fr" : "Congo", + "nl" : "Congo" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "CG" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Congo, The Democratic Republic of the", + "fr" : "Congo, The Democratic Republic of the", + "nl" : "Congo, The Democratic Republic of the" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "CD" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Cook Islands", + "fr" : "Cook Islands", + "nl" : "Cook Islands" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "CK" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Costa Rica", + "fr" : "Costa Rica", + "nl" : "Costa Rica" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "CR" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Cote D'Ivoire", + "fr" : "Cote D'Ivoire", + "nl" : "Cote D'Ivoire" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "CI" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Croatia", + "fr" : "Croatia", + "nl" : "Croatia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "HR" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Cuba", + "fr" : "Cuba", + "nl" : "Cuba" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "CU" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Cyprus", + "fr" : "Cyprus", + "nl" : "Cyprus" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "CY" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Czech Republic", + "fr" : "Czech Republic", + "nl" : "Czech Republic" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "CZ" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Denmark", + "fr" : "Denmark", + "nl" : "Denmark" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "DK" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Djibouti", + "fr" : "Djibouti", + "nl" : "Djibouti" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "DJ" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Dominica", + "fr" : "Dominica", + "nl" : "Dominica" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "DM" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Dominican Republic", + "fr" : "Dominican Republic", + "nl" : "Dominican Republic" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "DO" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Ecuador", + "fr" : "Ecuador", + "nl" : "Ecuador" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "EC" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Egypt", + "fr" : "Egypt", + "nl" : "Egypt" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "EG" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "El Salvador", + "fr" : "El Salvador", + "nl" : "El Salvador" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "SV" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Equatorial Guinea", + "fr" : "Equatorial Guinea", + "nl" : "Equatorial Guinea" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "GQ" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Eritrea", + "fr" : "Eritrea", + "nl" : "Eritrea" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "ER" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Estonia", + "fr" : "Estonia", + "nl" : "Estonia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "EE" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Ethiopia", + "fr" : "Ethiopia", + "nl" : "Ethiopia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "ET" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Falkland Islands (Malvinas)", + "fr" : "Falkland Islands (Malvinas)", + "nl" : "Falkland Islands (Malvinas)" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "FK" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Faroe Islands", + "fr" : "Faroe Islands", + "nl" : "Faroe Islands" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "FO" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Fiji", + "fr" : "Fiji", + "nl" : "Fiji" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "FJ" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Finland", + "fr" : "Finland", + "nl" : "Finland" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "FI" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "France", + "fr" : "France", + "nl" : "France" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "FR" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "French Guiana", + "fr" : "French Guiana", + "nl" : "French Guiana" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "GF" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "French Polynesia", + "fr" : "French Polynesia", + "nl" : "French Polynesia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "PF" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "French Southern Territories", + "fr" : "French Southern Territories", + "nl" : "French Southern Territories" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "TF" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Gabon", + "fr" : "Gabon", + "nl" : "Gabon" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "GA" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Gambia", + "fr" : "Gambia", + "nl" : "Gambia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "GM" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Georgia", + "fr" : "Georgia", + "nl" : "Georgia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "GE" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Germany", + "fr" : "Germany", + "nl" : "Germany" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "DE" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Ghana", + "fr" : "Ghana", + "nl" : "Ghana" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "GH" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Gibraltar", + "fr" : "Gibraltar", + "nl" : "Gibraltar" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "GI" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Greece", + "fr" : "Greece", + "nl" : "Greece" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "GR" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Greenland", + "fr" : "Greenland", + "nl" : "Greenland" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "GL" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Grenada", + "fr" : "Grenada", + "nl" : "Grenada" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "GD" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Guadeloupe", + "fr" : "Guadeloupe", + "nl" : "Guadeloupe" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "GP" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Guam", + "fr" : "Guam", + "nl" : "Guam" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "GU" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Guatemala", + "fr" : "Guatemala", + "nl" : "Guatemala" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "GT" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Guernsey", + "fr" : "Guernsey", + "nl" : "Guernsey" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "GG" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Guinea", + "fr" : "Guinea", + "nl" : "Guinea" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "GN" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Guinea-Bissau", + "fr" : "Guinea-Bissau", + "nl" : "Guinea-Bissau" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "GW" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Guyana", + "fr" : "Guyana", + "nl" : "Guyana" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "GY" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Haiti", + "fr" : "Haiti", + "nl" : "Haiti" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "HT" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Heard Island and Mcdonald Islands", + "fr" : "Heard Island and Mcdonald Islands", + "nl" : "Heard Island and Mcdonald Islands" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "HM" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Holy See (Vatican City State)", + "fr" : "Holy See (Vatican City State)", + "nl" : "Holy See (Vatican City State)" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "VA" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Honduras", + "fr" : "Honduras", + "nl" : "Honduras" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "HN" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Hong Kong", + "fr" : "Hong Kong", + "nl" : "Hong Kong" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "HK" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Hungary", + "fr" : "Hungary", + "nl" : "Hungary" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "HU" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Iceland", + "fr" : "Iceland", + "nl" : "Iceland" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "IS" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "India", + "fr" : "India", + "nl" : "India" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "IN" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Indonesia", + "fr" : "Indonesia", + "nl" : "Indonesia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "ID" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Iran, Islamic Republic Of", + "fr" : "Iran, Islamic Republic Of", + "nl" : "Iran, Islamic Republic Of" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "IR" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Iraq", + "fr" : "Iraq", + "nl" : "Iraq" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "IQ" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Ireland", + "fr" : "Ireland", + "nl" : "Ireland" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "IE" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Isle of Man", + "fr" : "Isle of Man", + "nl" : "Isle of Man" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "IM" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Israel", + "fr" : "Israel", + "nl" : "Israel" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "IL" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Italy", + "fr" : "Italy", + "nl" : "Italy" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "IT" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Jamaica", + "fr" : "Jamaica", + "nl" : "Jamaica" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "JM" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Japan", + "fr" : "Japan", + "nl" : "Japan" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "JP" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Jersey", + "fr" : "Jersey", + "nl" : "Jersey" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "JE" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Jordan", + "fr" : "Jordan", + "nl" : "Jordan" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "JO" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Kazakhstan", + "fr" : "Kazakhstan", + "nl" : "Kazakhstan" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "KZ" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Kenya", + "fr" : "Kenya", + "nl" : "Kenya" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "KE" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Kiribati", + "fr" : "Kiribati", + "nl" : "Kiribati" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "KI" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Korea, Democratic People'S Republic of", + "fr" : "Korea, Democratic People'S Republic of", + "nl" : "Korea, Democratic People'S Republic of" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "KP" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Korea, Republic of", + "fr" : "Korea, Republic of", + "nl" : "Korea, Republic of" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "KR" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Kuwait", + "fr" : "Kuwait", + "nl" : "Kuwait" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "KW" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Kyrgyzstan", + "fr" : "Kyrgyzstan", + "nl" : "Kyrgyzstan" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "KG" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Lao People'S Democratic Republic", + "fr" : "Lao People'S Democratic Republic", + "nl" : "Lao People'S Democratic Republic" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "LA" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Latvia", + "fr" : "Latvia", + "nl" : "Latvia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "LV" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Lebanon", + "fr" : "Lebanon", + "nl" : "Lebanon" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "LB" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Lesotho", + "fr" : "Lesotho", + "nl" : "Lesotho" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "LS" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Liberia", + "fr" : "Liberia", + "nl" : "Liberia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "LR" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Libyan Arab Jamahiriya", + "fr" : "Libyan Arab Jamahiriya", + "nl" : "Libyan Arab Jamahiriya" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "LY" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Liechtenstein", + "fr" : "Liechtenstein", + "nl" : "Liechtenstein" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "LI" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Lithuania", + "fr" : "Lithuania", + "nl" : "Lithuania" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "LT" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Luxembourg", + "fr" : "Luxembourg", + "nl" : "Luxembourg" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "LU" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Macao", + "fr" : "Macao", + "nl" : "Macao" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "MO" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Macedonia, The Former Yugoslav Republic of", + "fr" : "Macedonia, The Former Yugoslav Republic of", + "nl" : "Macedonia, The Former Yugoslav Republic of" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "MK" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Madagascar", + "fr" : "Madagascar", + "nl" : "Madagascar" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "MG" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Malawi", + "fr" : "Malawi", + "nl" : "Malawi" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "MW" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Malaysia", + "fr" : "Malaysia", + "nl" : "Malaysia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "MY" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Maldives", + "fr" : "Maldives", + "nl" : "Maldives" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "MV" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Mali", + "fr" : "Mali", + "nl" : "Mali" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "ML" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Malta", + "fr" : "Malta", + "nl" : "Malta" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "MT" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Marshall Islands", + "fr" : "Marshall Islands", + "nl" : "Marshall Islands" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "MH" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Martinique", + "fr" : "Martinique", + "nl" : "Martinique" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "MQ" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Mauritania", + "fr" : "Mauritania", + "nl" : "Mauritania" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "MR" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Mauritius", + "fr" : "Mauritius", + "nl" : "Mauritius" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "MU" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Mayotte", + "fr" : "Mayotte", + "nl" : "Mayotte" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "YT" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Mexico", + "fr" : "Mexico", + "nl" : "Mexico" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "MX" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Micronesia, Federated States of", + "fr" : "Micronesia, Federated States of", + "nl" : "Micronesia, Federated States of" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "FM" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Moldova, Republic of", + "fr" : "Moldova, Republic of", + "nl" : "Moldova, Republic of" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "MD" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Monaco", + "fr" : "Monaco", + "nl" : "Monaco" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "MC" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Mongolia", + "fr" : "Mongolia", + "nl" : "Mongolia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "MN" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Montserrat", + "fr" : "Montserrat", + "nl" : "Montserrat" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "MS" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Morocco", + "fr" : "Morocco", + "nl" : "Morocco" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "MA" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Mozambique", + "fr" : "Mozambique", + "nl" : "Mozambique" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "MZ" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Myanmar", + "fr" : "Myanmar", + "nl" : "Myanmar" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "MM" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Namibia", + "fr" : "Namibia", + "nl" : "Namibia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "NA" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Nauru", + "fr" : "Nauru", + "nl" : "Nauru" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "NR" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Nepal", + "fr" : "Nepal", + "nl" : "Nepal" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "NP" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Netherlands", + "fr" : "Netherlands", + "nl" : "Netherlands" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "NL" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Netherlands Antilles", + "fr" : "Netherlands Antilles", + "nl" : "Netherlands Antilles" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "AN" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "New Caledonia", + "fr" : "New Caledonia", + "nl" : "New Caledonia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "NC" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "New Zealand", + "fr" : "New Zealand", + "nl" : "New Zealand" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "NZ" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Nicaragua", + "fr" : "Nicaragua", + "nl" : "Nicaragua" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "NI" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Niger", + "fr" : "Niger", + "nl" : "Niger" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "NE" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Nigeria", + "fr" : "Nigeria", + "nl" : "Nigeria" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "NG" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Niue", + "fr" : "Niue", + "nl" : "Niue" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "NU" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Norfolk Island", + "fr" : "Norfolk Island", + "nl" : "Norfolk Island" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "NF" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Northern Mariana Islands", + "fr" : "Northern Mariana Islands", + "nl" : "Northern Mariana Islands" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "MP" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Norway", + "fr" : "Norway", + "nl" : "Norway" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "NO" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Oman", + "fr" : "Oman", + "nl" : "Oman" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "OM" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Pakistan", + "fr" : "Pakistan", + "nl" : "Pakistan" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "PK" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Palau", + "fr" : "Palau", + "nl" : "Palau" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "PW" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Palestinian Territory, Occupied", + "fr" : "Palestinian Territory, Occupied", + "nl" : "Palestinian Territory, Occupied" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "PS" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Panama", + "fr" : "Panama", + "nl" : "Panama" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "PA" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Papua New Guinea", + "fr" : "Papua New Guinea", + "nl" : "Papua New Guinea" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "PG" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Paraguay", + "fr" : "Paraguay", + "nl" : "Paraguay" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "PY" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Peru", + "fr" : "Peru", + "nl" : "Peru" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "PE" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Philippines", + "fr" : "Philippines", + "nl" : "Philippines" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "PH" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Pitcairn", + "fr" : "Pitcairn", + "nl" : "Pitcairn" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "PN" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Poland", + "fr" : "Poland", + "nl" : "Poland" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "PL" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Portugal", + "fr" : "Portugal", + "nl" : "Portugal" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "PT" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Puerto Rico", + "fr" : "Puerto Rico", + "nl" : "Puerto Rico" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "PR" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Qatar", + "fr" : "Qatar", + "nl" : "Qatar" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "QA" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Reunion", + "fr" : "Reunion", + "nl" : "Reunion" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "RE" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Romania", + "fr" : "Romania", + "nl" : "Romania" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "RO" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Russian Federation", + "fr" : "Russian Federation", + "nl" : "Russian Federation" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "RU" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "RWANDA", + "fr" : "RWANDA", + "nl" : "RWANDA" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "RW" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Saint Helena", + "fr" : "Saint Helena", + "nl" : "Saint Helena" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "SH" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Saint Kitts and Nevis", + "fr" : "Saint Kitts and Nevis", + "nl" : "Saint Kitts and Nevis" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "KN" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Saint Lucia", + "fr" : "Saint Lucia", + "nl" : "Saint Lucia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "LC" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Saint Pierre and Miquelon", + "fr" : "Saint Pierre and Miquelon", + "nl" : "Saint Pierre and Miquelon" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "PM" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Saint Vincent and the Grenadines", + "fr" : "Saint Vincent and the Grenadines", + "nl" : "Saint Vincent and the Grenadines" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "VC" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Samoa", + "fr" : "Samoa", + "nl" : "Samoa" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "WS" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "San Marino", + "fr" : "San Marino", + "nl" : "San Marino" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "SM" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Sao Tome and Principe", + "fr" : "Sao Tome and Principe", + "nl" : "Sao Tome and Principe" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "ST" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Saudi Arabia", + "fr" : "Saudi Arabia", + "nl" : "Saudi Arabia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "SA" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Senegal", + "fr" : "Senegal", + "nl" : "Senegal" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "SN" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Serbia and Montenegro", + "fr" : "Serbia and Montenegro", + "nl" : "Serbia and Montenegro" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "CS" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Seychelles", + "fr" : "Seychelles", + "nl" : "Seychelles" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "SC" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Sierra Leone", + "fr" : "Sierra Leone", + "nl" : "Sierra Leone" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "SL" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Singapore", + "fr" : "Singapore", + "nl" : "Singapore" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "SG" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Slovakia", + "fr" : "Slovakia", + "nl" : "Slovakia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "SK" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Slovenia", + "fr" : "Slovenia", + "nl" : "Slovenia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "SI" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Solomon Islands", + "fr" : "Solomon Islands", + "nl" : "Solomon Islands" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "SB" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Somalia", + "fr" : "Somalia", + "nl" : "Somalia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "SO" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "South Africa", + "fr" : "South Africa", + "nl" : "South Africa" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "ZA" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "South Georgia and the South Sandwich Islands", + "fr" : "South Georgia and the South Sandwich Islands", + "nl" : "South Georgia and the South Sandwich Islands" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "GS" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Spain", + "fr" : "Spain", + "nl" : "Spain" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "ES" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Sri Lanka", + "fr" : "Sri Lanka", + "nl" : "Sri Lanka" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "LK" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Sudan", + "fr" : "Sudan", + "nl" : "Sudan" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "SD" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Suriname", + "fr" : "Suriname", + "nl" : "Suriname" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "SR" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Svalbard and Jan Mayen", + "fr" : "Svalbard and Jan Mayen", + "nl" : "Svalbard and Jan Mayen" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "SJ" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Swaziland", + "fr" : "Swaziland", + "nl" : "Swaziland" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "SZ" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Sweden", + "fr" : "Sweden", + "nl" : "Sweden" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "SE" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Switzerland", + "fr" : "Switzerland", + "nl" : "Switzerland" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "CH" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Syrian Arab Republic", + "fr" : "Syrian Arab Republic", + "nl" : "Syrian Arab Republic" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "SY" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Taiwan, Province of China", + "fr" : "Taiwan, Province of China", + "nl" : "Taiwan, Province of China" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "TW" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Tajikistan", + "fr" : "Tajikistan", + "nl" : "Tajikistan" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "TJ" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Tanzania, United Republic of", + "fr" : "Tanzania, United Republic of", + "nl" : "Tanzania, United Republic of" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "TZ" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Thailand", + "fr" : "Thailand", + "nl" : "Thailand" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "TH" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Timor-Leste", + "fr" : "Timor-Leste", + "nl" : "Timor-Leste" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "TL" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Togo", + "fr" : "Togo", + "nl" : "Togo" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "TG" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Tokelau", + "fr" : "Tokelau", + "nl" : "Tokelau" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "TK" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Tonga", + "fr" : "Tonga", + "nl" : "Tonga" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "TO" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Trinidad and Tobago", + "fr" : "Trinidad and Tobago", + "nl" : "Trinidad and Tobago" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "TT" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Tunisia", + "fr" : "Tunisia", + "nl" : "Tunisia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "TN" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Turkey", + "fr" : "Turkey", + "nl" : "Turkey" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "TR" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Turkmenistan", + "fr" : "Turkmenistan", + "nl" : "Turkmenistan" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "TM" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Turks and Caicos Islands", + "fr" : "Turks and Caicos Islands", + "nl" : "Turks and Caicos Islands" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "TC" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Tuvalu", + "fr" : "Tuvalu", + "nl" : "Tuvalu" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "TV" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Uganda", + "fr" : "Uganda", + "nl" : "Uganda" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "UG" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Ukraine", + "fr" : "Ukraine", + "nl" : "Ukraine" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "UA" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "United Arab Emirates", + "fr" : "United Arab Emirates", + "nl" : "United Arab Emirates" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "AE" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "United Kingdom", + "fr" : "United Kingdom", + "nl" : "United Kingdom" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "GB" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "United States", + "fr" : "United States", + "nl" : "United States" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "US" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "United States Minor Outlying Islands", + "fr" : "United States Minor Outlying Islands", + "nl" : "United States Minor Outlying Islands" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "UM" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Uruguay", + "fr" : "Uruguay", + "nl" : "Uruguay" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "UY" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Uzbekistan", + "fr" : "Uzbekistan", + "nl" : "Uzbekistan" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "UZ" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Vanuatu", + "fr" : "Vanuatu", + "nl" : "Vanuatu" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "VU" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Venezuela", + "fr" : "Venezuela", + "nl" : "Venezuela" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "VE" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Viet Nam", + "fr" : "Viet Nam", + "nl" : "Viet Nam" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "VN" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Virgin Islands, British", + "fr" : "Virgin Islands, British", + "nl" : "Virgin Islands, British" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "VG" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Virgin Islands, U.S.", + "fr" : "Virgin Islands, U.S.", + "nl" : "Virgin Islands, U.S." + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "VI" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Wallis and Futuna", + "fr" : "Wallis and Futuna", + "nl" : "Wallis and Futuna" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "WF" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Western Sahara", + "fr" : "Western Sahara", + "nl" : "Western Sahara" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "EH" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Yemen", + "fr" : "Yemen", + "nl" : "Yemen" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "YE" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Zambia", + "fr" : "Zambia", + "nl" : "Zambia" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "ZM" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + { + "label" : "BE", + "position" : NumberInt(1), + "type" : "Object", + "index" : NumberInt(0), + "content" : { + "name" : { + "label" : "name", + "position" : NumberInt(1), + "type" : "input", + "index" : NumberInt(0), + "content" : { + "en" : "Zimbabwe", + "fr" : "Zimbabwe", + "nl" : "Zimbabwe" + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "labelMatch" : false, + "valueMatch" : [ + + ] + }, + "code" : { + "label" : "code", + "position" : NumberInt(2), + "type" : "value", + "index" : NumberInt(1), + "content" : { + + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "value" : "ZW" + } + }, + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "value", + "key" : "code" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + } + ], + "toComplete" : false, + "locked" : false, + "lockedValue" : false, + "open" : true, + "_NEW" : { + "type" : "Object", + "key" : "FR" + }, + "labelMatch" : false, + "valueMatch" : [ + + ] + } + }, + "createdAt" : ISODate("2020-08-20T14:10:34.221+0000"), + "d" : [ + "admin", + "L3iCtdd4sogcaRucTWmUI634ij53" + ], + "label" : "Liste des pays", + "r" : [ + "public", + "L3iCtdd4sogcaRucTWmUI634ij53" + ], + "title" : "countryCodes", + "updatedAt" : ISODate("2020-08-20T14:21:55.663+0000"), + "w" : [ + "admin", + "L3iCtdd4sogcaRucTWmUI634ij53" + ], + "position" : null +} \ No newline at end of file diff --git a/services/module-cms/index.ts b/services/module-cms/index.ts new file mode 100644 index 0000000..f82b7b7 --- /dev/null +++ b/services/module-cms/index.ts @@ -0,0 +1,19 @@ +import 'reflect-metadata'; +import CategoryResolver from '@services/module-cms/functions/categories/category.resolver'; +import CategoryAdminResolver from '@services/module-cms/functions/categories/category.resolver.admin'; +import EmailSettingsAdminResolver from '@services/module-cms/functions/emails-settings/emails-settings.resolver'; +import ProjectSettingsAdminResolver from './functions/project-settings/project-settings.resolver.admin'; +import ListResolver from './functions/lists/list.resolver'; +import ListAdminResolver from './functions/lists/list.resolver.admin'; +import { PageAdminResolvers } from './functions/pages/pages.resolver.admin'; +import ErrorSettingsAdminResolver from './functions/error-settings/errors-settings.resolver'; + +export const ApiCMSAppResolvers = [CategoryResolver, ListResolver]; +export const ApiCMSAdminResolvers = [ + ...PageAdminResolvers, + EmailSettingsAdminResolver, + CategoryAdminResolver, + ProjectSettingsAdminResolver, + ListAdminResolver, + ErrorSettingsAdminResolver, +]; diff --git a/services/module-cms/services/CmsService.ts b/services/module-cms/services/CmsService.ts new file mode 100644 index 0000000..5c16fa0 --- /dev/null +++ b/services/module-cms/services/CmsService.ts @@ -0,0 +1,66 @@ +import { FunctionResponse, fError, fSuccess } from '@seed/interfaces/response'; + +import CMS from '@services/module-cms/functions/pages/pages.model'; + +export default class CMSHelper { + private static instance: CMSHelper; + public EMAILS!: any; + + // eslint-disable-next-line @typescript-eslint/no-empty-function + private constructor() {} + + public static async getEmails(emailTitle: string, language: string): Promise> { + if (!this.instance) this.instance = new CMSHelper(); + if (!this.instance.EMAILS) { + const model = new CMS(); + + const page = await (await model.db()).findOne({ title: 'emails' }); + if (!page) return fError('emailsNotInitializedInCMS'); + + this.instance.EMAILS = page; + } + + if (!this.instance.EMAILS.content[emailTitle] || !this.instance.EMAILS.content[emailTitle].content) { + console.error('emailDataTemplate'); + return fError('emailDataNotFound'); + } + + const emailData = this.instance.EMAILS.content[emailTitle].content; + + const result = { + template: emailData.template ? (emailData.template.value ? emailData.template.value : 'default') : 'default', + subject: emailData.subject ? (emailData.subject.content[language] ? emailData.subject.content[language] : '') : '', + // subtitle: emailData.subtitle ? (emailData.subtitle.content[language] ? emailData.subtitle.content[language] : '') : '', + body: emailData.body ? (emailData.body.content[language] ? emailData.body.content[language] : '') : '', + // button: emailData.button ? (emailData.button.content[language] ? emailData.button.content[language] : undefined) : undefined, + // buttonLink: emailData.buttonLink + // ? emailData.buttonLink.content[language] + // ? emailData.buttonLink.content[language] + // : undefined + // : undefined, + cci: emailData.cci ? (emailData.cci.value ? emailData.cci.value.split(',') : undefined) : undefined, + fromEmail: emailData.fromEmail ? (emailData.fromEmail.value ? emailData.fromEmail.value : '') : '', + replyToEmail: emailData.replyToEmail ? (emailData.replyToEmail.value ? emailData.replyToEmail.value : '') : '', + }; + return fSuccess(result); + } + public static async getEmailTemplate(name = 'default', language = 'en'): Promise> { + if (!this.instance) this.instance = new CMSHelper(); + if (!this.instance.EMAILS) { + const model = new CMS(); + + const page = await (await model.db()).findOne({ title: 'emails' }); + if (!page) return fError('emailsNotInitializedInCMS'); + + this.instance.EMAILS = page; + } + + try { + const template = this.instance.EMAILS.content.template.content[name].content[language]; + return fSuccess(template); + } catch (error) { + console.error('emailTemplateNotFound'); + return fError('emailTemplateNotFound'); + } + } +} diff --git a/services/module-cms/services/PageService.ts b/services/module-cms/services/PageService.ts new file mode 100644 index 0000000..8661060 --- /dev/null +++ b/services/module-cms/services/PageService.ts @@ -0,0 +1,32 @@ +import { promiseAll } from '@seed/helpers/Utils'; +import { uploadToS3 } from '@services/api-uploads/services/S3Service'; +import { AvailableTranslation, TranslatableComponent } from '@src/__components/components'; +import TreatmentContent from './TreatmentContent'; + +export const updateCaching = async (model: any) => { + console.log('Getting all the pages'); + const pages = await model.getAll({ query: {} }); + + const languages = Object.keys(AvailableTranslation); + + const promises: any[] = []; + + const treatment = new TreatmentContent(pages, { languages }); + const contentTest = treatment.getAllTreatments(); + + for (const key in contentTest) { + const contentLanguage = contentTest[key]; + promises.push( + uploadToS3({ + acl: false, + contentType: 'application/json', + buffer: Buffer.from(JSON.stringify(contentLanguage)), + fileName: `content/${process.env.NODE_ENV}-${key}.json`, + }), + ); + } + + await promiseAll(promises); + + console.log('Done uploading to S3'); +}; diff --git a/services/module-cms/services/TreatmentContent.ts b/services/module-cms/services/TreatmentContent.ts new file mode 100644 index 0000000..eff2fb2 --- /dev/null +++ b/services/module-cms/services/TreatmentContent.ts @@ -0,0 +1,172 @@ +/* eslint-disable @typescript-eslint/explicit-function-return-type */ +import cloneDeep from 'lodash/cloneDeep'; + +interface OptionProps { + languages: string[]; + whitelistedContentKeys?: string[]; + whitelistedCmsPages?: string[]; + blacklistedCmsPages?: string[]; + keyNewContent?: string; + arrayAsObject?: boolean; +} + +class TreatmentContent { + pages: any[]; + + options: any = { + languages: ['en', 'fr', 'nl'], + whitelistedContentKeys: ['position', 'label', 'link', 'url', 'value', 'type', 'src'], + keyNewContent: '_NEW', + arrayAsObject: true, + }; + + constructor(pages: any[], options?: OptionProps) { + // make sure to have json + for (let i = 0; i < pages.length; i += 1) { + const item = pages[i]; + item.content = typeof item.content === 'string' ? JSON.parse(item.content) : item.content; + } + this.pages = pages; + + this.options = { + ...this.options, + ...options, + }; + } + + treatmentContent = (content: object, parentContent: any, parentType: string | undefined, lg?: string): void => { + const keys = Object.keys(content); + keys.forEach((key): void => { + if (key === this.options.keyNewContent) return; + + const elem = content[key]; + const currentContent = elem.content || elem; + + if (!currentContent) return; + + if (elem.type === 'Object') { + // Mode where we keep list as array + if (!this.options.arrayAsObject && parentType && parentType === 'Array') { + const newObj = {}; + parentContent.push(newObj); + this.treatmentContent(currentContent, newObj, undefined, lg); + } else { + let label = key; + // Mode where list are handled as object + if (parentType && parentType === 'Array' && elem.label) { + label = elem.label; + } + parentContent[label] = {}; + if (elem.position) { + parentContent[label].position = parseInt(elem.position, 10); + } + this.treatmentContent(currentContent, parentContent[label], undefined, lg); + } + } else if (elem.type === 'Array') { + if (!this.options.arrayAsObject) { + parentContent[key] = []; + } else { + parentContent[key] = {}; + } + + this.treatmentContent(currentContent, parentContent[key], 'Array', lg); + } else if (elem.type === 'img') { + parentContent[key] = { + src: elem.src, + }; + } else { + const newObj = parentContent[key] ? cloneDeep(parentContent[key]) : {}; + + const subKeys = Object.keys(elem); + subKeys.forEach((subKey): void => { + if (['content', 'alt', 'label'].indexOf(subKey) < 0) { + if (this.options.whitelistedContentKeys.indexOf(subKey) < 0) return; + + newObj[subKey] = elem[subKey]; + // parse number + if (elem[subKey] && typeof elem[subKey] === 'number') { + newObj[subKey] = parseInt(elem[subKey], 10); + } + } else if (subKey === 'label') { + if (parentType === 'Array' && !newObj.value) { + newObj.value = elem[subKey]; + } + if (this.options.whitelistedContentKeys.indexOf(subKey) < 0) return; + + newObj[subKey] = elem[subKey]; + } + }); + // attach content and alt (if present) + const langKeys = Object.keys(currentContent); + langKeys.forEach((langKey): void => { + if (lg && langKey.indexOf(lg) < 0) return; + + newObj[langKey] = currentContent[langKey]; + }); + // attach alt object to image + if (elem.alt) { + newObj.alt = {}; + const altKeys = Object.keys(currentContent); + altKeys.forEach((altKey): void => { + if (lg && altKey.indexOf(lg) < 0) return; + + newObj[altKey] = currentContent[altKey]; + }); + } + + // Attaching new content to the parent + if (parentType && parentType === 'Array') { + if (!this.options.arrayAsObject) { + parentContent.push(newObj); + } else { + const label = newObj.label || key; + parentContent[label] = newObj; + } + } else { + parentContent[newObj.dataKey || key] = newObj; + } + } + }); + }; + + treatmentPage = (page: any, language?: string) => { + const newContent = {}; + + this.treatmentContent(page, newContent, undefined, language); + + return newContent; + }; + + public treatmentPages = (targetLg?: string) => { + const output = {}; + + for (let i = 0; i < this.pages.length; i += 1) { + const item = this.pages[i]; + + const key = item.key || item.title || item.label; + + if (this.options.whitelistedCmsPages && this.options.whitelistedCmsPages.indexOf(key) < 0) continue; + + if (this.options.blacklistedCmsPages && this.options.blacklistedCmsPages.indexOf(key) >= 0) continue; + + if (item.content) output[key] = this.treatmentPage(item.content, targetLg); + } + + return output; + }; + + getAllTreatments = () => { + const output: any = {}; + + for (let i = 0; i < this.options.languages.length; i += 1) { + const lg = this.options.languages[i]; + + output[lg] = this.treatmentPages(lg); + } + output.all = this.treatmentPages(); + + return output; + }; +} + +export default TreatmentContent; diff --git a/services/module-comments/.gitignore b/services/module-comments/.gitignore new file mode 100644 index 0000000..9bea433 --- /dev/null +++ b/services/module-comments/.gitignore @@ -0,0 +1,2 @@ + +.DS_Store diff --git a/services/module-comments/README.md b/services/module-comments/README.md new file mode 100644 index 0000000..4d632ed --- /dev/null +++ b/services/module-comments/README.md @@ -0,0 +1,14 @@ +# Make-it Comments Service + +[![](https://ci6.googleusercontent.com/proxy/2HHmNs0zn0uGJb9RqML73pX6Bmd-BntMaIYR6IpXeIHKqn2_tG60C1pLNtMHoLHOk1I5CT8k5x9gvb-C1zuudHnD5HDOpvHljVYcKmDL--rb0Ar8IT4UB7YaG1c=s0-d-e1-ft#https://makeit-assets.s3.eu-central-1.amazonaws.com/signature_sanawar.png)](https://makeit-studio.com) + +![Build Status](https://travis-ci.org/joemccann/dillinger.svg?branch=master) + +### Documentation + +https://docs.google.com/document/d/1Nw-9-QFOXBXZxrxHPCv6vojtwedWN2hVsihQr-NIza4 + +### Todo + +- Commenting Stats +- Likes, etc diff --git a/services/module-comments/__tests/playground.gql b/services/module-comments/__tests/playground.gql new file mode 100644 index 0000000..00d8f67 --- /dev/null +++ b/services/module-comments/__tests/playground.gql @@ -0,0 +1,13 @@ +mutation addOneReview { + addOneReview(input: { + ressourceId:"1111", + ressourceModel:ArticleModel, + note: 1, + comment: "Test mode" + }) { + _id, + comment + note + createdAt + } +} \ No newline at end of file diff --git a/services/module-comments/functions/comments/comment.model.ts b/services/module-comments/functions/comments/comment.model.ts new file mode 100644 index 0000000..3f0c02a --- /dev/null +++ b/services/module-comments/functions/comments/comment.model.ts @@ -0,0 +1,47 @@ +import { ObjectType, Field } from 'type-graphql'; + +import { Permission } from '@seed/interfaces/permission'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { EngineModel } from '@lib/seed/engine/EngineModel'; +import { plainToClass } from 'class-transformer'; +import { CustomSearchEngine } from '@seed/services/database/DBRequestService'; +import { IEngineSchema } from '@seed/engine/EngineSchema'; +import { CommentDBIntefaceSchema, CommentDBSchema, CommentSchema } from './schemas/comments.schema'; +import { CommentArgsSchemma } from './schemas/comments.schema.input'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.public, AccountTypeEnum.admin], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class CommentModel extends EngineModel { + public constructor(input?: CommentDBIntefaceSchema & Partial) { + const dataInit = plainToClass(CommentDBSchema, input || {}); + super({ + // ...init, + collectionName: 'comments.reviews', + permissions: permissions, + dataInit, + }); + } + + @Field({ nullable: true }) + idk?: string; + + searchEngine(): CustomSearchEngine { + const sEngine: CustomSearchEngine = { + ressourceId: { + operation: '$eq', + }, + }; + + return sEngine; + } + + plainToClass(plain: any): CommentSchema | CommentSchema[] { + return plainToClass(CommentSchema, plain); + } +} diff --git a/services/module-comments/functions/comments/comment.service.ts b/services/module-comments/functions/comments/comment.service.ts new file mode 100644 index 0000000..14ad5cc --- /dev/null +++ b/services/module-comments/functions/comments/comment.service.ts @@ -0,0 +1,111 @@ +// import CommentModel, { NewCommentInput } from './comment.model'; +// import { ApolloContext } from '@seed/interfaces/context'; +// import { BaseGraphModel } from '@seed/graphql/BaseGraphModel'; +// import { addOneGeneric } from '@seed/graphql/BaseService'; + +import { GetArgs } from '@seed/graphql/Request'; +import { ApolloContext } from '@seed/interfaces/context'; +import { modelsLoaders } from '@src/__indexes/__loaders'; +import CommentModel from './comment.model'; +import { CommentSchema } from './schemas/comments.schema'; +import { CommentNewInputSchema } from './schemas/comments.schema.input'; + +export class CommentService { + static async getComments(ressourceId: string, pagination?: GetArgs): Promise { + try { + return new CommentModel().getMany({ query: { ressourceId }, pagination }); + } catch (error) { + throw error; + } + } + + static async addComment(input: CommentNewInputSchema, sentBy: string): Promise { + try { + // Check if the loader for that ressource exist + let modelGeneric = modelsLoaders[input.ressourceModel]; + if (modelGeneric) { + try { + await modelGeneric.getOne({ query: { _id: input.ressourceId } }); + await modelGeneric.updateOneCustom({ query: { _id: input.ressourceId }, updateRequest: { $inc: { totalComment: 1 } } }); + } catch (error) { + console.error(error); + } + } + + // if (input.parentId) { + // const commentModel = await new VideoCommentModel().getOne({ _id: input.parentId }); + + // // Update count of reply + // await commentModel.updateOne({ _id: commentModel._id }, { totalReplies: (commentModel.totalReplies || 0) + 1 }); + // } + + const newComment = new CommentModel({ + ...input, + sentBy, + }); + + return await newComment.saveOne({}); + } catch (error) { + throw error; + } + } +} + +// @Mutation(() => VideoCommentModel) +// @Authorized() +// async deleteVideoComment(@Arg('input') input: DeleteVideoCommentInput, @Ctx() ctx: ApolloContext): Promise { +// try { +// const comment = await new VideoCommentModel().getOne({ _id: input.commentId }); + +// if (comment.userId != ctx.ctx.user._id) throw new ApolloError('notAllowed', '400'); + +// // Update count of comments +// const videoModel = await new VideoModel().getOne({ _id: comment.videoId }); +// await videoModel.updateOne({ _id: videoModel._id }, { totalComment: (videoModel.totalComment || 1) - 1 }); + +// if (comment.parentId) { +// // Update count of reply +// await comment.updateOne({ _id: comment.parentId }, { totalReplies: (comment.totalReplies || 1) - 1 }); +// } + +// return deleteOneGeneric(comment, input.commentId); +// } catch (error) { +// throw error; +// } + +// @Mutation(() => CommentModel) +// @Authorized() +// async editComment(@Arg('id') id: string, @Arg('input') input: EditCommentInput, @Ctx() ctx: ApolloContext): Promise { +// const model = new CommentModel(); +// const message = await model.getOne({ _id: id }); + +// if (message.sentBy !== ctx.ctx.user._id) +// throw new ApolloError('permissions.notAllowed', '400', { you: ctx.ctx.user._id, allowed: message.sentBy }); + +// return model.updateOne( +// { _id: id }, +// { +// message: input.message, +// edited: true, +// updatedAt: new Date(), +// }, +// ); +// } + +// @Mutation(() => CommentModel) +// @Authorized() +// async deleteComment(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { +// const model = new CommentModel(); +// const message = await model.getOne({ _id: id }); + +// if (message.sentBy !== ctx.ctx.user._id) +// throw new ApolloError('permissions.notAllowed', '400', { you: ctx.ctx.user._id, allowed: message.sentBy }); + +// return model.updateOne( +// { _id: id }, +// { +// deleted: true, +// updatedAt: new Date(), +// }, +// ); +// } diff --git a/services/module-comments/functions/comments/commentStat.model.ts b/services/module-comments/functions/comments/commentStat.model.ts new file mode 100644 index 0000000..118f1b6 --- /dev/null +++ b/services/module-comments/functions/comments/commentStat.model.ts @@ -0,0 +1,60 @@ +import { ObjectType, Field, ID, ArgsType, InputType, Int } from 'type-graphql'; + +import { BaseGraphModel } from '@seed/graphql/BaseGraphModel'; +import ImageComponent from '@seed/interfaces/components'; +import { GetArgs } from '@seed/graphql/Request'; + +import { Permission } from '@seed/interfaces/permission'; +import { AccountTypeEnum } from '@src/accounts/account.components'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.public], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export class CommentReadBy { + @Field() + accountId: string; + + @Field() + readAt: Date; +} + +@ObjectType() +export default class CommentStatModel extends BaseGraphModel { + public constructor() { + super({ + // ...init, + collectionName: 'stats', + permissions: permissions, + }); + } + // eslint-disable-next-line @typescript-eslint/no-unused-vars + @Field(() => ID) + readonly _id: string; + + @Field() + ressourceId: string; + + @Field() + totalComment: string; + + @Field() + lastCommented: Date; + + searchOptions(): string[] { + return []; + } + filterOptions(): string[] { + return []; + } +} + +@ArgsType() +export class CommentArgs { + @Field(() => GetArgs, { nullable: true }) + pagination?: GetArgs; +} diff --git a/services/module-comments/functions/comments/resolvers/comment.resolver.ts b/services/module-comments/functions/comments/resolvers/comment.resolver.ts new file mode 100644 index 0000000..78dda76 --- /dev/null +++ b/services/module-comments/functions/comments/resolvers/comment.resolver.ts @@ -0,0 +1,53 @@ +import { Resolver } from 'type-graphql'; +import { createEngineMutationResolver, createEngineQueryResolver } from '@seed/engine/genericResolvers/BaseEngineResolver'; + +import { AccountTypeEnum } from '@src/accounts/account.components'; + +import { CommentSchema } from '../schemas/comments.schema'; +import { CommentArgsSchemma, CommentNewInputSchema, CommentEditInputSchema } from '../schemas/comments.schema.input'; +import CommentModel from '../comment.model'; + +const CommentGenericQueryResolver = createEngineQueryResolver({ + domain: 'comments', + schemaName: CommentSchema, + modelName: CommentModel, + argsType: CommentArgsSchemma, + engineMiddleware: {}, +}); + +@Resolver(CommentSchema) +export class CommentQueryResolver extends CommentGenericQueryResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ +} + +const CommentGenericMutationResolver = createEngineMutationResolver({ + domain: 'comments', + schemaName: CommentSchema, + modelName: CommentModel, + newInput: CommentNewInputSchema, + editInput: CommentEditInputSchema, + engineMiddleware: { + authorization: [AccountTypeEnum.user], + }, +}); + +@Resolver(CommentSchema) +export class CommentMutationResolver extends CommentGenericMutationResolver { + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ +} + +export const CommentResolvers = [CommentQueryResolver, CommentMutationResolver]; diff --git a/services/module-comments/functions/comments/schemas/comments.schema.input.ts b/services/module-comments/functions/comments/schemas/comments.schema.input.ts new file mode 100644 index 0000000..06fbdb1 --- /dev/null +++ b/services/module-comments/functions/comments/schemas/comments.schema.input.ts @@ -0,0 +1,34 @@ +import ImageComponent from '@lib/seed/interfaces/components'; +import { ArgsType, Field, InputType, Int } from 'type-graphql'; +import { CommentReviewInputSchema } from '../../components'; +import { CommentSchema } from './comments.schema'; + +@InputType() +export class CommentNewInputSchema extends CommentReviewInputSchema implements Partial { + @Field() + comment: string; +} + +@InputType() +export class CommentNewInputCustomSchema implements Partial { + @Field() + comment: string; + + @Field(() => ImageComponent, { nullable: true }) + file?: ImageComponent; +} + +@InputType() +export class CommentEditInputSchema implements Partial { + @Field({ nullable: true }) + comment?: string; + + @Field(() => ImageComponent, { nullable: true }) + file?: ImageComponent; +} + +@ArgsType() +export class CommentArgsSchemma { + @Field() + ressourceId: string; +} diff --git a/services/module-comments/functions/comments/schemas/comments.schema.ts b/services/module-comments/functions/comments/schemas/comments.schema.ts new file mode 100644 index 0000000..7613515 --- /dev/null +++ b/services/module-comments/functions/comments/schemas/comments.schema.ts @@ -0,0 +1,25 @@ +import { ObjectType, Field, ArgsType, InputType, Int } from 'type-graphql'; + +import { IEngineSchema, MetaBy, MetaPermissions } from '@lib/seed/engine/EngineSchema'; +import { IsNotEmpty } from 'class-validator'; +import { CommentReviewDBBaseSchema } from '../../components'; + +@ObjectType() +export class CommentDBIntefaceSchema extends CommentReviewDBBaseSchema { + @Field() + @IsNotEmpty() + comment: string; +} + +@ObjectType() +export class CommentDBSchema extends CommentDBIntefaceSchema {} + +@ObjectType({ implements: IEngineSchema }) +export class CommentSchema extends CommentDBSchema implements IEngineSchema { + _id: string; + organisationId?: string | undefined; + by?: MetaBy | undefined; + permissions: MetaPermissions; + createdAt: Date; + updatedAt: Date; +} diff --git a/services/module-comments/functions/components.ts b/services/module-comments/functions/components.ts new file mode 100644 index 0000000..9c43022 --- /dev/null +++ b/services/module-comments/functions/components.ts @@ -0,0 +1,83 @@ +/* + ███████╗███╗ ██╗██╗ ██╗███╗ ███╗ + ██╔════╝████╗ ██║██║ ██║████╗ ████║ + █████╗ ██╔██╗ ██║██║ ██║██╔████╔██║ + ██╔══╝ ██║╚██╗██║██║ ██║██║╚██╔╝██║ + ███████╗██║ ╚████║╚██████╔╝██║ ╚═╝ ██║ + ╚══════╝╚═╝ ╚═══╝ ╚═════╝ ╚═╝ ╚═╝ +*/ + +import { IsRefExist } from '@lib/seed/engine/decorators/db.guard'; +import ImageComponent from '@lib/seed/interfaces/components'; +import { CommentModelEnum } from '@src/__components/components'; +import { IsNotEmpty } from 'class-validator'; +import { InputType, ObjectType, Field, Int, UseMiddleware, ArgsType } from 'type-graphql'; + +/* + + ██████╗ ██████╗ ███╗ ███╗██████╗ ██████╗ ███╗ ██╗███████╗███╗ ██╗████████╗███████╗ +██╔════╝██╔═══██╗████╗ ████║██╔══██╗██╔═══██╗████╗ ██║██╔════╝████╗ ██║╚══██╔══╝██╔════╝ +██║ ██║ ██║██╔████╔██║██████╔╝██║ ██║██╔██╗ ██║█████╗ ██╔██╗ ██║ ██║ ███████╗ +██║ ██║ ██║██║╚██╔╝██║██╔═══╝ ██║ ██║██║╚██╗██║██╔══╝ ██║╚██╗██║ ██║ ╚════██║ +╚██████╗╚██████╔╝██║ ╚═╝ ██║██║ ╚██████╔╝██║ ╚████║███████╗██║ ╚████║ ██║ ███████║ + ╚═════╝ ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═════╝ ╚═╝ ╚═══╝╚══════╝╚═╝ ╚═══╝ ╚═╝ ╚══════╝ + +*/ + +@InputType() +@ObjectType() +export class ReportedInfo { + @Field() + accountId: string; + + @Field() + addedAt: Date; +} + +@InputType() +@ObjectType() +export class CommentReviewInputSchema { + @Field(() => CommentModelEnum) + @IsNotEmpty() + ressourceModel: CommentModelEnum; + + @Field() + @IsNotEmpty() + @IsRefExist() + ressourceId: string; + + @Field(() => ImageComponent, { nullable: true }) + file?: ImageComponent; +} + +@InputType() +@ObjectType() +export class CommentReviewDBBaseSchema { + @Field(() => Int, { nullable: true }) + note?: number; + + @Field({ nullable: true }) + comment?: string; + + @Field(() => ImageComponent, { nullable: true }) + file?: ImageComponent; + + @Field(() => [ReportedInfo], { nullable: true }) + reported?: ReportedInfo[]; + + @Field() + @IsNotEmpty() + sentBy: string; + + @Field({ nullable: true }) + edited?: boolean; + + @Field({ nullable: true }) + deleted?: boolean; + + @Field({ nullable: true }) + parentId?: string; + + @Field(() => Int, { nullable: true }) + numberReplies?: number; +} diff --git a/services/module-comments/functions/reviews/resolvers/reviews.resolver.ts b/services/module-comments/functions/reviews/resolvers/reviews.resolver.ts new file mode 100644 index 0000000..fbdf6e4 --- /dev/null +++ b/services/module-comments/functions/reviews/resolvers/reviews.resolver.ts @@ -0,0 +1,53 @@ +import { Resolver } from 'type-graphql'; + +import { createEngineMutationResolver, createEngineQueryResolver } from '@seed/engine/genericResolvers/BaseEngineResolver'; +import { AccountTypeEnum } from '@src/accounts/account.components'; + +import { ReviewSchema } from '../schemas/reviews.schema'; +import ReviewModel from '../reviews.model'; +import { ReviewArgsSchemma, ReviewEditInputSchema, ReviewNewInputSchema } from '../schemas/reviews.schema.input'; + +const ReviewGenericQueryResolver = createEngineQueryResolver({ + domain: 'reviews', + schemaName: ReviewSchema, + modelName: ReviewModel, + argsType: ReviewArgsSchemma, + engineMiddleware: {}, +}); + +@Resolver(ReviewSchema) +export class ReviewQueryResolver extends ReviewGenericQueryResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ +} + +const ReviewGenericMutationResolver = createEngineMutationResolver({ + domain: 'reviews', + schemaName: ReviewSchema, + modelName: ReviewModel, + newInput: ReviewNewInputSchema, + editInput: ReviewEditInputSchema, + engineMiddleware: { + authorization: [AccountTypeEnum.user], + }, +}); + +@Resolver(ReviewSchema) +export class ReviewMutationResolver extends ReviewGenericMutationResolver { + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ +} + +export const ReviewResolvers = [ReviewQueryResolver, ReviewMutationResolver]; diff --git a/services/module-comments/functions/reviews/reviews.model.ts b/services/module-comments/functions/reviews/reviews.model.ts new file mode 100644 index 0000000..81b7de4 --- /dev/null +++ b/services/module-comments/functions/reviews/reviews.model.ts @@ -0,0 +1,52 @@ +import { ObjectType } from 'type-graphql'; + +import { Permission } from '@seed/interfaces/permission'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { EngineModel } from '@lib/seed/engine/EngineModel'; +import { plainToClass } from 'class-transformer'; +import { CustomSearchEngine } from '@seed/services/database/DBRequestService'; +import { IEngineSchema } from '@seed/engine/EngineSchema'; +import { ReviewDBIntefaceSchema, ReviewDBSchema, ReviewSchema } from './schemas/reviews.schema'; +import { ReviewArgsSchemma } from './schemas/reviews.schema.input'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.public, AccountTypeEnum.admin], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class ReviewModel extends EngineModel { + public constructor(input?: ReviewDBIntefaceSchema & Partial) { + const dataInit = plainToClass(ReviewDBSchema, input || {}); + super({ + // ...init, + collectionName: 'comments.reviews', + permissions: permissions, + dataInit, + }); + } + + searchEngine(): CustomSearchEngine { + const sEngine: CustomSearchEngine = { + ressourceId: { + operation: '$eq', + }, + noteFilter: { + operation: 'between', + dbFName: 'note', + beetweenConfig: { + inputFromField: 'noteMin', + inputToField: 'noteMax', + }, + }, + }; + + return sEngine; + } + + plainToClass(plain: any): ReviewSchema | ReviewSchema[] { + return plainToClass(ReviewSchema, plain); + } +} diff --git a/services/module-comments/functions/reviews/schemas/reviews.schema.input.ts b/services/module-comments/functions/reviews/schemas/reviews.schema.input.ts new file mode 100644 index 0000000..d18600e --- /dev/null +++ b/services/module-comments/functions/reviews/schemas/reviews.schema.input.ts @@ -0,0 +1,36 @@ +import ImageComponent from '@lib/seed/interfaces/components'; +import { ArgsType, Field, InputType, Int } from 'type-graphql'; +import { CommentReviewInputSchema } from '../../components'; +import { ReviewSchema } from './reviews.schema'; + +@InputType() +export class ReviewNewInputSchema extends CommentReviewInputSchema implements Partial { + @Field(() => Int) + note: number; + + @Field({ nullable: true }) + comment?: string; +} + +@InputType() +export class ReviewEditInputSchema implements Partial { + @Field(() => Int, { nullable: true }) + note?: number; + + @Field({ nullable: true }) + comment?: string; + + @Field(() => ImageComponent, { nullable: true }) + file?: ImageComponent; +} + +@ArgsType() +export class ReviewArgsSchemma { + @Field() + ressourceId: string; + + @Field(() => Int, { nullable: true }) + noteMin?: number; + @Field(() => Int, { nullable: true }) + noteMax?: number; +} diff --git a/services/module-comments/functions/reviews/schemas/reviews.schema.ts b/services/module-comments/functions/reviews/schemas/reviews.schema.ts new file mode 100644 index 0000000..3712f15 --- /dev/null +++ b/services/module-comments/functions/reviews/schemas/reviews.schema.ts @@ -0,0 +1,23 @@ +import { ObjectType, Field, ArgsType, InputType, Int } from 'type-graphql'; + +import { IEngineSchema, MetaBy, MetaPermissions } from '@lib/seed/engine/EngineSchema'; +import { IsNotEmpty } from 'class-validator'; +import { CommentReviewDBBaseSchema } from '../../components'; + +export class ReviewDBIntefaceSchema extends CommentReviewDBBaseSchema { + @Field(() => Int) + @IsNotEmpty() + note: number; +} + +export class ReviewDBSchema extends ReviewDBIntefaceSchema {} + +@ObjectType({ implements: IEngineSchema }) +export class ReviewSchema extends ReviewDBSchema implements IEngineSchema { + _id: string; + organisationId?: string | undefined; + by?: MetaBy | undefined; + permissions: MetaPermissions; + createdAt: Date; + updatedAt: Date; +} diff --git a/services/module-favorites/.gitignore b/services/module-favorites/.gitignore new file mode 100644 index 0000000..9bea433 --- /dev/null +++ b/services/module-favorites/.gitignore @@ -0,0 +1,2 @@ + +.DS_Store diff --git a/services/module-favorites/README.md b/services/module-favorites/README.md new file mode 100644 index 0000000..3811eaa --- /dev/null +++ b/services/module-favorites/README.md @@ -0,0 +1,34 @@ +# module-favorites + +Need to add this to the account Model + + @Field(() => [FavoriteComponent], { nullable: 'itemsAndList' }) + favLikes?: FavoriteComponent[]; + + +Need to add the corresponding resolvers +-- app + + AccountFavResolver, + AccountLikeResolver + +-- admin + +Need to update the loaders & ressource enum + +export enum RessourceEnum { + 'NAME_OF_RESSOURCE' = 'NAME_OF_RESSOURCE', +} +registerEnumType(RessourceEnum, { + name: 'RessourceEnum', +}); + +export const modelsLoaders = { + accounts: new AccountModel(), + NAME_OF_RESSOURCE: new NAME_OF_RESSOURCE_MODEL(), +}; + + + + + diff --git a/services/module-favorites/__tests/favorites.playground.graphql b/services/module-favorites/__tests/favorites.playground.graphql new file mode 100644 index 0000000..c38d2c5 --- /dev/null +++ b/services/module-favorites/__tests/favorites.playground.graphql @@ -0,0 +1,7 @@ +query me {me{favLikes{ressourceId,ressourceType,addedAt,favorite,liked}}} + +mutation accountsFavoritesAddOne{accountsFavoritesAddOne(input:{ressourceId:"7d9c6ffb-a1c2-4e5d-9488-663725a1c730",ressourceType:products}){ressourceId,ressourceType,addedAt,favorite}} +mutation accountsFavoritesRemoveOne{accountsFavoritesRemoveOne(input:{ressourceId:"7d9c6ffb-a1c2-4e5d-9488-663725a1c730"}){success}} +mutation accountsLikesAddOne{accountsLikesAddOne(input:{ressourceId:"7d9c6ffb-a1c2-4e5d-9488-663725a1c730",ressourceType:products}){ressourceId,ressourceType,addedAt,favorite}} +mutation accountsDislikesAddOne{accountsDislikesAddOne(input:{ressourceId:"7d9c6ffb-a1c2-4e5d-9488-663725a1c730",ressourceType:products}){ressourceId,ressourceType,addedAt,favorite}} +mutation accountsLikesRemoveOne{accountsLikesRemoveOne(input:{ressourceId:"7d9c6ffb-a1c2-4e5d-9488-663725a1c730"}){success}} \ No newline at end of file diff --git a/services/module-favorites/components.ts b/services/module-favorites/components.ts new file mode 100644 index 0000000..6f7abe9 --- /dev/null +++ b/services/module-favorites/components.ts @@ -0,0 +1,54 @@ +import { ObjectType, Field, InputType } from 'type-graphql'; +import { RessourceEnum } from '@src/__components/components'; + +/* + ███████╗███╗ ██╗██╗ ██╗███╗ ███╗ + ██╔════╝████╗ ██║██║ ██║████╗ ████║ + █████╗ ██╔██╗ ██║██║ ██║██╔████╔██║ + ██╔══╝ ██║╚██╗██║██║ ██║██║╚██╔╝██║ + ███████╗██║ ╚████║╚██████╔╝██║ ╚═╝ ██║ + ╚══════╝╚═╝ ╚═══╝ ╚═════╝ ╚═╝ ╚═╝ +*/ + +/* + + ██████╗ ██████╗ ███╗ ███╗██████╗ ██████╗ ███╗ ██╗███████╗███╗ ██╗████████╗███████╗ +██╔════╝██╔═══██╗████╗ ████║██╔══██╗██╔═══██╗████╗ ██║██╔════╝████╗ ██║╚══██╔══╝██╔════╝ +██║ ██║ ██║██╔████╔██║██████╔╝██║ ██║██╔██╗ ██║█████╗ ██╔██╗ ██║ ██║ ███████╗ +██║ ██║ ██║██║╚██╔╝██║██╔═══╝ ██║ ██║██║╚██╗██║██╔══╝ ██║╚██╗██║ ██║ ╚════██║ +╚██████╗╚██████╔╝██║ ╚═╝ ██║██║ ╚██████╔╝██║ ╚████║███████╗██║ ╚████║ ██║ ███████║ + ╚═════╝ ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═════╝ ╚═╝ ╚═══╝╚══════╝╚═╝ ╚═══╝ ╚═╝ ╚══════╝ + +*/ + +@ObjectType() +export class FavoriteComponent { + @Field() + ressourceId: string; + + @Field(() => RessourceEnum) + ressourceType: RessourceEnum; + + @Field() + addedAt: Date; + + @Field({ nullable: true }) + favorite?: boolean; + @Field({ nullable: true }) + liked?: boolean; +} + +@InputType() +export class AddInput { + @Field() + ressourceId: string; + + @Field(() => RessourceEnum) + ressourceType: RessourceEnum; +} + +@InputType() +export class RemoveInput { + @Field() + ressourceId: string; +} diff --git a/services/module-favorites/functions/favLikes/favLikes.resolver.admin.ts b/services/module-favorites/functions/favLikes/favLikes.resolver.admin.ts new file mode 100644 index 0000000..f3f8ffe --- /dev/null +++ b/services/module-favorites/functions/favLikes/favLikes.resolver.admin.ts @@ -0,0 +1,14 @@ +import AccountModel from '@src/accounts/account.model'; +import { Resolver } from 'type-graphql'; + +@Resolver(AccountModel) +export default class AccountFavLikeAdminResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ +} diff --git a/services/module-favorites/functions/favLikes/favLikes.resolver.ts b/services/module-favorites/functions/favLikes/favLikes.resolver.ts new file mode 100644 index 0000000..2040cec --- /dev/null +++ b/services/module-favorites/functions/favLikes/favLikes.resolver.ts @@ -0,0 +1,211 @@ +import { Resolver, Arg, Mutation, Ctx, Authorized, FieldResolver, Root } from 'type-graphql'; +import AccountModel from '@src/accounts/account.model'; +import { ApolloContext } from '@lib/seed/interfaces/context'; +import { AddInput, FavoriteComponent, RemoveInput } from '@services/module-favorites/components'; +import { ApolloError } from 'apollo-server-lambda'; +import { modelsLoaders } from '@src/__indexes/__loaders'; +import { BaseGraphModel } from '@lib/seed/graphql/BaseGraphModel'; +import _ from 'lodash'; +import { SuccessResponse } from '@lib/seed/interfaces/response'; +import { oneToManyComplexity } from '@lib/seed/graphql/Middleware'; + +@Resolver(AccountModel) +export class AccountFavResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ + + @FieldResolver(() => [FavoriteComponent], { complexity: oneToManyComplexity }) + async getFavorites(@Root() account: AccountModel, @Ctx() ctx: ApolloContext): Promise<(FavoriteComponent | Error)[]> { + const favLikes = account.favLikes || []; + return _.filter(favLikes, { favorite: true }); + } + + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + + @Mutation(() => FavoriteComponent) + @Authorized() + async accountsFavoritesAddOne(@Arg('input') input: AddInput, @Ctx() ctx: ApolloContext): Promise { + const currentAccount = ctx.ctx.user; + // Check if element exists + if (!modelsLoaders[input.ressourceType]) throw new ApolloError('badConfiguration', '400'); + + const ressourceModel = modelsLoaders[input.ressourceType]; + await ressourceModel.getOne({ _id: input.ressourceId }, ctx); + + const elementToSave: FavoriteComponent = { + ...input, + favorite: true, + addedAt: new Date(), + }; + + const favLikes: FavoriteComponent[] = currentAccount.favLikes || []; + + // Check if already in the sub array + const index = _.findIndex(favLikes, { ressourceId: input.ressourceId }); + if (index === -1) favLikes.push(elementToSave); + else { + favLikes[index].favorite = true; + } + + await currentAccount.updateOne({ _id: currentAccount._id }, { favLikes: favLikes }, ctx); + + return elementToSave; + } + + @Mutation(() => SuccessResponse) + @Authorized() + async accountsFavoritesRemoveOne(@Arg('input') input: RemoveInput, @Ctx() ctx: ApolloContext): Promise { + const currentAccount = ctx.ctx.user; + const favLikes: FavoriteComponent[] = currentAccount.favLikes || []; + + // Check if already in the sub array + const index = _.findIndex(favLikes, { ressourceId: input.ressourceId }); + if (index === -1) return { success: true }; + else { + // Check needs to remove entirely or just from the field + if (favLikes[index].liked === undefined) favLikes.splice(index, 1); + else favLikes[index].favorite = undefined; + } + + await currentAccount.updateOne({ _id: currentAccount._id }, { favLikes: favLikes }, ctx); + + return { success: true }; + } +} + +@Resolver(AccountModel) +export class AccountLikeResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ + + @FieldResolver(() => [FavoriteComponent], { complexity: oneToManyComplexity }) + async getLiked(@Root() account: AccountModel, @Ctx() ctx: ApolloContext): Promise<(FavoriteComponent | Error)[]> { + const favLikes = account.favLikes || []; + return _.filter(favLikes, { liked: true }); + } + + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + + @Mutation(() => FavoriteComponent) + @Authorized() + async accountsLikesAddOne(@Arg('input') input: AddInput, @Ctx() ctx: ApolloContext): Promise { + const currentAccount = ctx.ctx.user; + // Check if element exists + if (!modelsLoaders[input.ressourceType]) throw new ApolloError('badConfiguration', '400'); + + const ressourceModel = modelsLoaders[input.ressourceType]; + await ressourceModel.getOne({ _id: input.ressourceId }, ctx); + + const elementToSave: FavoriteComponent = { + ...input, + liked: true, + addedAt: new Date(), + }; + + const favLikes: FavoriteComponent[] = currentAccount.favLikes || []; + + // Check if already in the sub array + const index = _.findIndex(favLikes, { ressourceId: input.ressourceId }); + if (index === -1) favLikes.push(elementToSave); + else { + favLikes[index].liked = true; + } + + await currentAccount.updateOne({ _id: currentAccount._id }, { favLikes: favLikes }, ctx); + + return elementToSave; + } + + @Mutation(() => FavoriteComponent) + @Authorized() + async accountsDislikesAddOne(@Arg('input') input: AddInput, @Ctx() ctx: ApolloContext): Promise { + const currentAccount = ctx.ctx.user; + // Check if element exists + if (!modelsLoaders[input.ressourceType]) throw new ApolloError('badConfiguration', '400'); + + const ressourceModel = modelsLoaders[input.ressourceType]; + await ressourceModel.getOne({ _id: input.ressourceId }, ctx); + + const elementToSave: FavoriteComponent = { + ...input, + liked: false, + addedAt: new Date(), + }; + + const favLikes: FavoriteComponent[] = currentAccount.favLikes || []; + + // Check if already in the sub array + const index = _.findIndex(favLikes, { ressourceId: input.ressourceId }); + if (index === -1) favLikes.push(elementToSave); + else { + favLikes[index].liked = false; + } + + await currentAccount.updateOne({ _id: currentAccount._id }, { favLikes: favLikes }, ctx); + + return elementToSave; + } + + @Mutation(() => SuccessResponse) + @Authorized() + async accountsLikesRemoveOne(@Arg('input') input: RemoveInput, @Ctx() ctx: ApolloContext): Promise { + const currentAccount = ctx.ctx.user; + const favLikes: FavoriteComponent[] = currentAccount.favLikes || []; + + // Check if already in the sub array + const index = _.findIndex(favLikes, { ressourceId: input.ressourceId }); + if (index === -1) return { success: true }; + else { + // Check needs to remove entirely or just from the field + if (favLikes[index].favorite === undefined) favLikes.splice(index, 1); + else favLikes[index].liked = undefined; + } + + await currentAccount.updateOne({ _id: currentAccount._id }, { favLikes: favLikes }, ctx); + + return { success: true }; + } +} diff --git a/services/module-payments/.gitignore b/services/module-payments/.gitignore new file mode 100644 index 0000000..9bea433 --- /dev/null +++ b/services/module-payments/.gitignore @@ -0,0 +1,2 @@ + +.DS_Store diff --git a/services/module-payments/README.md b/services/module-payments/README.md new file mode 100644 index 0000000..717d94a --- /dev/null +++ b/services/module-payments/README.md @@ -0,0 +1,28 @@ +# module-payments + +Need to add this to the account Model + + @Field(() => AccountPaymentInfo, { nullable: true }) + paymentInfo?: AccountPaymentInfo; + +Need to add the corresponding resolvers +-- app + + ...ModulePaymentAppResolver + +-- admin + +Stripe Install + + - Install the env in DB + +Stripe config + +Url to add in the stripe webhook + +-- Custom domain (recommended) +https://xxx-api.example.com/payments/webhook + +Once you have setup a webhook, you will received a signature key whsec_XXXX that you need to put as STRIPE_HOOK_SECRET in your env + + diff --git a/services/module-payments/__examples/checkout.resolver.ts b/services/module-payments/__examples/checkout.resolver.ts new file mode 100644 index 0000000..0817d7b --- /dev/null +++ b/services/module-payments/__examples/checkout.resolver.ts @@ -0,0 +1,46 @@ +import { Resolver, Authorized, Ctx } from 'type-graphql'; +import { Mutation, Arg } from 'type-graphql'; +import { ApolloContext } from '@seed/interfaces/context'; + +// From Module +import { OrderEngineSchema } from '../functions/orders/schemas/order.schema'; +import { CheckoutEngineInput } from '../functions/orders/schemas/order.schema.input'; + +// From SRC + +// EXAMPLES + +// @Resolver(OrderModel) +// export default class CheckoutEngineResolver { +// /* +// ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ +// ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ +// ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ +// ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ +// ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ +// ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ +// */ + +// /* +// ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ +// ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ +// ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ +// ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ +// ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ +// ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ +// */ + +// async cartsCheckout( +// @Arg('input') input: CheckoutEngineInput, +// @Arg('paymentType', { nullable: true }) paymentType: PaymentProvider, +// @Ctx() ctx: ApolloContext, +// ): Promise { +// try { +// const pt = paymentType || 'stripe'; + +// return await createCheckout({ payment: pt, contactInfo: input, input, ctx }); +// } catch (error) { +// throw error; +// } +// } +// } diff --git a/services/module-payments/__examples/env.import.json b/services/module-payments/__examples/env.import.json new file mode 100644 index 0000000..3ee62fa --- /dev/null +++ b/services/module-payments/__examples/env.import.json @@ -0,0 +1,102 @@ +{ + "_id" : "8c4c28fb-d89d-434c-a962-f4c43dbda68d", + "key" : "STRIPE_SKEY", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin", + ], + "w" : [ + "admin", + ], + "d" : [ + "admin", + ], +}, +{ + "_id" : "dec94533-a457-4c66-9ead-4e754243ec89", + "key" : "STRIPE_PUBLIC_KEY", + "value" :"", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin", + ], + "w" : [ + "admin", + ], + "d" : [ + "admin", + ], +}, +{ + "_id" : "870d1674-367a-4937-9d66-d36e889d4953", + "key" : "STRIPE_CURRENCY", + "value" :"usd", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin", + ], + "w" : [ + "admin", + ], + "d" : [ + "admin", + ], +}, +{ + "_id" : "5ed19459-1b4b-462b-88fb-5fbb2d698229", + "key" : "STRIPE_ORDERDESCRIPTION", + "value" :"You order from our webplatform", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin", + ], + "w" : [ + "admin", + ], + "d" : [ + "admin", + ], +}, +{ + "_id" : "5ed19459-1b4b-462b-88fb-5fxx2d698229", + "key" : "STRIPE_ACCOUNT_REFRESH_URL", + "value" :"https://makeit-studio.com", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin", + ], + "w" : [ + "admin", + ], + "d" : [ + "admin", + ], +}, +{ + "_id" : "5edxx459-1b4b-462b-88fb-5fxx2d698229", + "key" : "STRIPE_ACCOUNT_RETURN_URL", + "value" :"https://makeit-studio.com", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin", + ], + "w" : [ + "admin", + ], + "d" : [ + "admin", + ], +}, \ No newline at end of file diff --git a/services/module-payments/__examples/stripe.env.json b/services/module-payments/__examples/stripe.env.json new file mode 100644 index 0000000..ead0408 --- /dev/null +++ b/services/module-payments/__examples/stripe.env.json @@ -0,0 +1,85 @@ +{ + "_id" : "5ed19459-1b4b-462b-88fb-5fbb2d698229", + "key" : "STRIPE_ORDERDESCRIPTION", + "value" : "order from our webplatform", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin" + ], + "w" : [ + "admin" + ], + "d" : [ + "admin" + ] +} +{ + "_id" : "870d1674-367a-4937-9d66-d36e889d4953", + "key" : "STRIPE_CURRENCY", + "value" : "", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin" + ], + "w" : [ + "admin" + ], + "d" : [ + "admin" + ] +} +{ + "_id" : "dec94533-a457-4c66-9ead-4e754243ec89", + "key" : "STRIPE_PUBLIC_KEY", + "value" : "", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin" + ], + "w" : [ + "admin" + ], + "d" : [ + "admin" + ] +} +{ + "_id" : "8c4c28fb-d89d-434c-a962-f4c43dbda68d", + "key" : "STRIPE_SKEY", + "value" : "", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin" + ], + "w" : [ + "admin" + ], + "d" : [ + "admin" + ] +} +{ + "_id" : "8c4c28fb-xx-434c-xx-f4c43dbda68d", + "key" : "STRIPE_HOOK_SECRET", + "value" : "", + "type" : "env", + "createdAt" : ISODate(), + "updatedAt" : ISODate(), + "r" : [ + "admin" + ], + "w" : [ + "admin" + ], + "d" : [ + "admin" + ] +} \ No newline at end of file diff --git a/services/module-payments/__handlers/stripeHookHandler.ts b/services/module-payments/__handlers/stripeHookHandler.ts new file mode 100644 index 0000000..f56fe60 --- /dev/null +++ b/services/module-payments/__handlers/stripeHookHandler.ts @@ -0,0 +1,102 @@ +import 'reflect-metadata'; +import axios from 'axios'; +import serverless from 'serverless-http'; +import express from 'express'; + +import { OrderEngineDBSchema } from '@services/module-payments/functions/orders/schemas/order.schema'; +import { SettingsCache } from '@lib/seed/graphql/Settings'; + +import StripeService from '@services/module-payments/services/stripe/StripeService'; +import { OrderStatusEnum } from '@services/module-payments/components/components.orders'; +import { PaymentStatusEnum } from '@services/module-payments/components/components.payments'; +import DB from '@seed/services/database/DBService'; + +import {afterPaymentSuccess, afterPaymentFailed } from '@src/orders/services/order.service' + + +const app = express(); + +app.use(require('body-parser').text({ type: '*/*' })); +app.post('/payments/webhook', async function (request, response) { + const settingsI = SettingsCache.getInstance(); + await settingsI.refreshCache(); + + + const body = request.body; + const signature = request.headers['stripe-signature']; + + const endpointSecret = process.env.STRIPE_HOOK_SECRET || ''; + + let event; + + const stripeInstance = await StripeService.getInstance(); + + try { + event = stripeInstance.stripe.webhooks.constructEvent(body, signature as any, endpointSecret); + } catch (err) { + console.error('[STRIPE] Hooks - ERROR', err); + response.status(400).end(); + return; + } + + const intent = event.data.object; + console.log('[STRIPE] Hooks - event', event['type'], 'intent', intent.id); + + const orderData = await (await DB.getInstance()).db + .collection('shop.orders') + .findOne({ 'paymentIntent.stripePaymentIntentData.id': intent.id }); + + if (!orderData) { + console.error('[STRIPE] Hooks - ERROR', 'no order found') + return; + } + + switch (event['type']) { + case 'payment_intent.succeeded': + // Get the order + try { + // Update the payment Intent and it's status + const dataToSave: Partial = { + paymentStatus: PaymentStatusEnum.paid, + orderStatus: OrderStatusEnum.processing, + }; + + // check if organisations ids on product + // const providersData: ProviderSchema[] = []; + // await setProviderData(orderData, await createApolloContext(), providersData); + + // if (providersData.length > 0) dataToSave.providerOrderItems = providersData; + + await (await DB.getInstance()).db.collection('shop.orders').updateOne({ _id: orderData._id }, { $set: dataToSave }); + + await afterPaymentSuccess(orderData); + } catch (error) { + console.error(error); + } + break; + case 'payment_intent.payment_failed': + const message = intent?.last_payment_error && intent?.last_payment_error.message; + console.log('Failed:', intent.id, message); + + // // Get the order + try { + // Update the payment Intent and it's status + const dataToSave: Partial = { + paymentStatus: undefined, + orderStatus: OrderStatusEnum.draft, + }; + await (await DB.getInstance()).db.collection('shop.orders').updateOne({ _id: orderData._id }, { $set: dataToSave }); + + await afterPaymentFailed(orderData); + } catch (error) { + console.error(error); + } + break; + } + + response.sendStatus(200); +}); + +module.exports.handler = serverless(app); + + diff --git a/services/module-payments/__tests/billingInfo.playground.graphql b/services/module-payments/__tests/billingInfo.playground.graphql new file mode 100644 index 0000000..428487a --- /dev/null +++ b/services/module-payments/__tests/billingInfo.playground.graphql @@ -0,0 +1,30 @@ +query me {me{paymentInfo{stripeInfo{customerId},paymentMethods{sPayMethodId,default,nameOnCard},billingInfos{firstName,lastName,vatNumber}}}} + + +mutation addOneBillingInfo { + accountsBillingInfosAddOne( + input:{ + firstName:"Sanawar",lastName:"Syed",email:"sanawar@makeit-studio.com",default:true, + address: {number:"11", zip:"1000", street: "Rue du college",city:"Bruxelles",country:"Belgique"} + }) + + {firstName,lastName,email} +} + +mutation editOneBillingInfo { + accountsBillingInfosEditOne( + id:"50f8e025-8645-4af5-b262-8d961ad3dca5", + input:{ + firstName:"Sanawar2",lastName:"Syed2",email:"sanawar@makeit-studio.com",default:true, + address: {number:"11", zip:"1000", street: "Rue du college",city:"Bruxelles",country:"Belgique"} + }) + + {firstName,lastName,email} +} + + +mutation deleteOneBillingInfo { + accountsBillingInfosDeleteOne( + id:"ca3b9ad2-4c78-419d-9bb5-079d6aeeccb5" + ) +} diff --git a/services/module-payments/__tests/pm.playground.graphql b/services/module-payments/__tests/pm.playground.graphql new file mode 100644 index 0000000..46c110e --- /dev/null +++ b/services/module-payments/__tests/pm.playground.graphql @@ -0,0 +1,18 @@ +query me {me{paymentInfo{stripeInfo{customerId},paymentMethods{sPayMethodId,default,nameOnCard},billingInfos{firstName,lastName,vatNumber}}}} + +mutation addOnePaymentMethod { + accountsPaymentMethodsAddOne( + input:{ + sPayMethodId:"pm_card_visa", + nameOnCard:"SANAWAR SYED AZOR ALI", + }) + + {nameOnCard, cardInfo{brand,country,exp_year,exp_month,last4}} +} + + +mutation deleteOnePaymentMethod { + accountsPaymentMethodsDeleteOne( + id:"4bf3e82a-af7e-4425-86d9-a84ab79a44fe" + ) +} \ No newline at end of file diff --git a/services/module-payments/components/components.errors.ts b/services/module-payments/components/components.errors.ts new file mode 100644 index 0000000..956b4c1 --- /dev/null +++ b/services/module-payments/components/components.errors.ts @@ -0,0 +1,8 @@ +export const enginePaymentErrorConfig = { + /* Payments error */ + 6000: 'There was an error with the payment provider', + 6001: 'You have no account registered with the payment provider', + + 6003: 'There are no lines in your cart', + 6004: 'There was an error with the paiement intent', +}; diff --git a/services/module-payments/components/components.orders.ts b/services/module-payments/components/components.orders.ts new file mode 100644 index 0000000..89a59d0 --- /dev/null +++ b/services/module-payments/components/components.orders.ts @@ -0,0 +1,85 @@ +import { registerEnumType, ObjectType, Field, InputType, Int } from 'type-graphql'; +import { PaymentStatusEnum } from './components.payments'; +import { AvailableTranslation } from '@src/__components/components'; + +/* + ███████╗███╗ ██╗██╗ ██╗███╗ ███╗ + ██╔════╝████╗ ██║██║ ██║████╗ ████║ + █████╗ ██╔██╗ ██║██║ ██║██╔████╔██║ + ██╔══╝ ██║╚██╗██║██║ ██║██║╚██╔╝██║ + ███████╗██║ ╚████║╚██████╔╝██║ ╚═╝ ██║ + ╚══════╝╚═╝ ╚═══╝ ╚═════╝ ╚═╝ ╚═╝ +*/ +export enum OrderTypeEnum { + online = 'online', + withShipping = 'withShipping', +} +registerEnumType(OrderTypeEnum, { + name: 'OrderType', +}); + +export enum OrderStatusEnum { + draft = 'draft', + processing = 'processing', + shipped = 'shipped', + completed = 'completed', + canceled = 'canceled', + refund = 'refund', +} +registerEnumType(OrderStatusEnum, { + name: 'OrderStatusEnum', +}); + +export enum OrderBusinessStatusEnum { + processing = 'processing', + shipped = 'shipped', + completed = 'completed', + canceled = 'canceled', + refund = 'refund', +} +registerEnumType(OrderBusinessStatusEnum, { + name: 'OrderBusinessStatusEnum', +}); + +/* + + ██████╗ ██████╗ ███╗ ███╗██████╗ ██████╗ ███╗ ██╗███████╗███╗ ██╗████████╗███████╗ +██╔════╝██╔═══██╗████╗ ████║██╔══██╗██╔═══██╗████╗ ██║██╔════╝████╗ ██║╚══██╔══╝██╔════╝ +██║ ██║ ██║██╔████╔██║██████╔╝██║ ██║██╔██╗ ██║█████╗ ██╔██╗ ██║ ██║ ███████╗ +██║ ██║ ██║██║╚██╔╝██║██╔═══╝ ██║ ██║██║╚██╗██║██╔══╝ ██║╚██╗██║ ██║ ╚════██║ +╚██████╗╚██████╔╝██║ ╚═╝ ██║██║ ╚██████╔╝██║ ╚████║███████╗██║ ╚████║ ██║ ███████║ + ╚═════╝ ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═════╝ ╚═╝ ╚═══╝╚══════╝╚═╝ ╚═══╝ ╚═╝ ╚══════╝ + +*/ + +@ObjectType() +@InputType('LocaleInfoInput') +export class LocaleInfo { + @Field({ nullable: true }) + countryCode?: string; + + @Field(() => AvailableTranslation, { nullable: true }) + locale?: AvailableTranslation; +} + +@ObjectType() +export class OrderInfoComponent { + @Field() + _id: string; + + @Field(() => OrderStatusEnum) + orderStatus: OrderStatusEnum; + + @Field(() => PaymentStatusEnum) + paymentStatus: PaymentStatusEnum; +} + +@ObjectType() +@InputType('AddOnInput') +export class AddOnComponent { + @Field() + addOnId: string; + + @Field() + quantity: number; +} diff --git a/services/module-payments/components/components.payments.ts b/services/module-payments/components/components.payments.ts new file mode 100644 index 0000000..18841d0 --- /dev/null +++ b/services/module-payments/components/components.payments.ts @@ -0,0 +1,205 @@ +import { registerEnumType, ObjectType, Field, InputType, Int } from 'type-graphql'; +import { StripePaymentIntentData } from './stripe.components'; +import { BillingAddressComponent } from './components'; + +/* + ███████╗███╗ ██╗██╗ ██╗███╗ ███╗ + ██╔════╝████╗ ██║██║ ██║████╗ ████║ + █████╗ ██╔██╗ ██║██║ ██║██╔████╔██║ + ██╔══╝ ██║╚██╗██║██║ ██║██║╚██╔╝██║ + ███████╗██║ ╚████║╚██████╔╝██║ ╚═╝ ██║ + ╚══════╝╚═╝ ╚═══╝ ╚═════╝ ╚═╝ ╚═╝ +*/ + +export enum cardTypeEnum { + 'alipay' = 'alipay', + 'au_becs_debit' = 'au_becs_debit', + 'bacs_debit' = 'bacs_debit', + 'bancontact' = 'bancontact', + 'card' = 'card', + 'eps' = 'eps', + 'fpx' = 'fpx', + 'giropay' = 'giropay', + 'ideal' = 'ideal', + 'p24' = 'p24', + 'sepa_debit' = 'sepa_debit', + 'sofort' = 'sofort', +} +registerEnumType(cardTypeEnum, { + name: 'cardTypeEnum', +}); + +export enum cardBrandEnum { + amex = 'amex', + diners = 'diners', + discover = 'discover', + jcb = 'jcb', + mastercard = 'mastercard', + unionpay = 'unionpay', + visa = 'visa', + unknown = 'unknown', +} +registerEnumType(cardBrandEnum, { + name: 'cardBrandEnum', +}); + +export enum PaymentStatusEnum { + pending = 'pending', + actionNeeded = 'actionNeeded', + paid = 'paid', + free = 'free', +} +registerEnumType(PaymentStatusEnum, { + name: 'PaymentStatusEnum', +}); + +export enum PaymentProvider { + free = 'free', + stripe = 'stripe', + bancontact = 'bancontact', + bankWire = 'bankWire', +} +registerEnumType(PaymentProvider, { + name: 'PaymentProvider', +}); + +export enum CurrencyEnum { + eur = 'eur', + usd = 'usd', +} +registerEnumType(CurrencyEnum, { + name: 'CurrencyEnum', +}); + +export enum InvoicingProvider { + directInvoice = 'directInvoice', +} +registerEnumType(InvoicingProvider, { + name: 'InvoicingProvider', +}); + +/* + + ██████╗ ██████╗ ███╗ ███╗██████╗ ██████╗ ███╗ ██╗███████╗███╗ ██╗████████╗███████╗ +██╔════╝██╔═══██╗████╗ ████║██╔══██╗██╔═══██╗████╗ ██║██╔════╝████╗ ██║╚══██╔══╝██╔════╝ +██║ ██║ ██║██╔████╔██║██████╔╝██║ ██║██╔██╗ ██║█████╗ ██╔██╗ ██║ ██║ ███████╗ +██║ ██║ ██║██║╚██╔╝██║██╔═══╝ ██║ ██║██║╚██╗██║██╔══╝ ██║╚██╗██║ ██║ ╚════██║ +╚██████╗╚██████╔╝██║ ╚═╝ ██║██║ ╚██████╔╝██║ ╚████║███████╗██║ ╚████║ ██║ ███████║ + ╚═════╝ ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═════╝ ╚═╝ ╚═══╝╚══════╝╚═╝ ╚═══╝ ╚═╝ ╚══════╝ + +*/ + +@ObjectType() +export class StripeInfo { + @Field() + customerId: string; +} + +@ObjectType() +export class StripePayoutInfo { + @Field() + accountId: string; + + @Field() + chargesEnabled: boolean; + @Field() + payoutsEnabled: boolean; + @Field() + detailsSubmitted: boolean; +} + +@ObjectType() +export class CardInfo { + @Field(() => cardBrandEnum) + brand: cardBrandEnum; + + @Field() + country: string; + @Field(() => Int) + exp_month: number; + @Field(() => Int) + exp_year: number; + @Field() + last4: string; +} + +@ObjectType() +export class PaymentMethodDetail { + @Field() + _id: string; + + @Field() + sPayMethodId: string; + + @Field() + default: boolean; + + @Field() + nameOnCard: string; + + @Field(() => cardTypeEnum) + type: cardTypeEnum; + + @Field(() => CardInfo, { nullable: true }) + cardInfo?: CardInfo; +} + +@InputType() +export class PaymentMethodInput { + @Field() + sPayMethodId: string; + + @Field() + nameOnCard: string; + + @Field({ nullable: true }) + default: boolean; +} + +@ObjectType() +export class AccountPaymentInfo { + @Field(() => StripeInfo, { nullable: true }) + stripeInfo?: StripeInfo; + + @Field(() => [PaymentMethodDetail], { nullable: true }) + paymentMethods?: PaymentMethodDetail[]; + + @Field(() => [BillingAddressComponent], { nullable: true }) + billingInfos?: BillingAddressComponent[]; +} + +@ObjectType() +export class AccountPayoutInfo { + @Field(() => StripePayoutInfo, { nullable: true }) + stripeInfo?: StripePayoutInfo; +} + +@ObjectType() +export class PaymentInfo { + @Field(() => PaymentProvider) + provider: PaymentProvider; + + @Field() + transactionId: string; +} + +@ObjectType() +export class PaymentIntentInfo { + @Field(() => PaymentProvider) + provider: PaymentProvider; + + @Field(() => StripePaymentIntentData) + stripePaymentIntentData?: StripePaymentIntentData; +} + +@ObjectType() +export class InvoiceInfo { + @Field(() => InvoicingProvider) + provider: InvoicingProvider; + + @Field() + invoiceId: string; + + @Field(() => Number) + invoiceNumber: number; +} diff --git a/services/module-payments/components/components.ts b/services/module-payments/components/components.ts new file mode 100644 index 0000000..a09a4b5 --- /dev/null +++ b/services/module-payments/components/components.ts @@ -0,0 +1,62 @@ +import { registerEnumType, ObjectType, Field, InputType, Int } from 'type-graphql'; +import { GraphQLJSONObject } from 'graphql-type-json'; +import { StripePaymentIntentData } from './stripe.components'; +import { AddressStrictComponent } from '@seed/interfaces/components.geo'; + +/* + ███████╗███╗ ██╗██╗ ██╗███╗ ███╗ + ██╔════╝████╗ ██║██║ ██║████╗ ████║ + █████╗ ██╔██╗ ██║██║ ██║██╔████╔██║ + ██╔══╝ ██║╚██╗██║██║ ██║██║╚██╔╝██║ + ███████╗██║ ╚████║╚██████╔╝██║ ╚═╝ ██║ + ╚══════╝╚═╝ ╚═══╝ ╚═════╝ ╚═╝ ╚═╝ +*/ + +/* + + ██████╗ ██████╗ ███╗ ███╗██████╗ ██████╗ ███╗ ██╗███████╗███╗ ██╗████████╗███████╗ +██╔════╝██╔═══██╗████╗ ████║██╔══██╗██╔═══██╗████╗ ██║██╔════╝████╗ ██║╚══██╔══╝██╔════╝ +██║ ██║ ██║██╔████╔██║██████╔╝██║ ██║██╔██╗ ██║█████╗ ██╔██╗ ██║ ██║ ███████╗ +██║ ██║ ██║██║╚██╔╝██║██╔═══╝ ██║ ██║██║╚██╗██║██╔══╝ ██║╚██╗██║ ██║ ╚════██║ +╚██████╗╚██████╔╝██║ ╚═╝ ██║██║ ╚██████╔╝██║ ╚████║███████╗██║ ╚████║ ██║ ███████║ + ╚═════╝ ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═════╝ ╚═╝ ╚═══╝╚══════╝╚═╝ ╚═══╝ ╚═╝ ╚══════╝ + +*/ + +@ObjectType() +@InputType('BillingAddressInput') +export class BillingAddressInput { + @Field() + firstName: string; + + @Field() + lastName: string; + + @Field(() => AddressStrictComponent) + address: AddressStrictComponent; + + @Field({ nullable: true }) + email?: string; + + @Field({ nullable: true }) + default?: boolean; + + @Field(() => GraphQLJSONObject, { nullable: true }) + customTags?: any; + + @Field({ nullable: true }) + company?: string; + + @Field({ nullable: true }) + vatNumber?: string; + + @Field({ nullable: true }) + fiscalForm?: string; +} + +@ObjectType() +@InputType('BillingAddress') +export class BillingAddressComponent extends BillingAddressInput { + @Field() + _id: string; +} diff --git a/services/module-payments/components/stripe.components.ts b/services/module-payments/components/stripe.components.ts new file mode 100644 index 0000000..299d908 --- /dev/null +++ b/services/module-payments/components/stripe.components.ts @@ -0,0 +1,48 @@ +import { Field, ObjectType } from 'type-graphql'; + +@ObjectType() +export class StripePaymentIntentData { + @Field() + id: string; + @Field({ nullable: true }) + client_secret?: string; + @Field() + currency: string; + @Field() + customer: string; + @Field() + status: string; + + // object: string; + // amount: number; + // amount_capturable: number; + // amount_received: number; + // application: any; + // application_fee_amount: any; + // canceled_at: any; + // cancellation_reason: any; + // capture_method: string; + // charges: Charges; + // confirmation_method: string; + // created: number; + + // description: any; + // invoice: any; + // last_payment_error: any; + // livemode: boolean; + // metadata: Metadata; + // next_action: any; + // on_behalf_of: any; + // payment_method: any; + // payment_method_options: PaymentMethodOptions; + // payment_method_types: string[]; + // receipt_email: any; + // review: any; + // setup_future_usage: any; + // shipping: any; + // source: any; + // statement_descriptor: any; + // statement_descriptor_suffix: any; + // transfer_data: any; + // transfer_group: any; +} diff --git a/services/module-payments/functions/billingInfos/billingInfo.resolver.admin.ts b/services/module-payments/functions/billingInfos/billingInfo.resolver.admin.ts new file mode 100644 index 0000000..34c1f93 --- /dev/null +++ b/services/module-payments/functions/billingInfos/billingInfo.resolver.admin.ts @@ -0,0 +1,14 @@ +import AccountModel from '@src/accounts/account.model'; +import { Resolver } from 'type-graphql'; + +@Resolver(AccountModel) +export default class AccountBillingInfoAdminResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ +} diff --git a/services/module-payments/functions/billingInfos/billingInfo.resolver.fields.ts b/services/module-payments/functions/billingInfos/billingInfo.resolver.fields.ts new file mode 100644 index 0000000..c1ad3c3 --- /dev/null +++ b/services/module-payments/functions/billingInfos/billingInfo.resolver.fields.ts @@ -0,0 +1,14 @@ +import AccountModel from '@src/accounts/account.model'; +import { Resolver } from 'type-graphql'; + +@Resolver(AccountModel) +export default class AccountBillingInfoFieldResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ +} diff --git a/services/module-payments/functions/billingInfos/billingInfo.resolver.ts b/services/module-payments/functions/billingInfos/billingInfo.resolver.ts new file mode 100644 index 0000000..e8a97d9 --- /dev/null +++ b/services/module-payments/functions/billingInfos/billingInfo.resolver.ts @@ -0,0 +1,75 @@ +import { Resolver, Query, FieldResolver, Root, Arg, Mutation, Ctx, Args, Authorized, ResolverInterface } from 'type-graphql'; +import AccountModel, { NewAccountInput } from '@src/accounts/account.model'; +import { ApolloContext } from '@lib/seed/interfaces/context'; +import { signinGeneric } from '@services/module-accounts/services/AccountService'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { BillingAddressComponent, BillingAddressInput } from '@services/module-payments/components/components'; + +@Resolver(AccountModel) +export default class AccountBillingInfoResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + + @Mutation(() => BillingAddressComponent) + @Authorized() + async accountsBillingInfosAddOne(@Arg('input') input: BillingAddressInput, @Ctx() ctx: ApolloContext): Promise { + const model = ctx.ctx.user; + + if (input.default && model.paymentInfo && model.paymentInfo.billingInfos) { + // Remove all default from others + await model.updateOneCustom( + { _id: model._id }, + { + $set: { 'paymentInfo.billingInfos.$[].default': false }, + }, + ctx, + ); + } else input.default = false; + + return await model.addOneSubRessource({ _id: model._id }, 'paymentInfo.billingInfos', input, ctx); + } + + @Mutation(() => BillingAddressComponent) + @Authorized() + async accountsBillingInfosEditOne( + @Arg('id') id: string, + @Arg('input') input: BillingAddressInput, + @Ctx() ctx: ApolloContext, + ): Promise { + const model = ctx.ctx.user; + + if (input.default) { + // Remove all default from others + await model.updateOneCustom( + { _id: model._id }, + { + $set: { 'paymentInfo.billingInfos.$[].default': false }, + }, + ctx, + ); + } else input.default = false; + + return await model.editOneSubRessource({ _id: model._id }, 'paymentInfo.billingInfos', id, input, ctx); + } + + @Mutation(() => String) + @Authorized() + async accountsBillingInfosDeleteOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = ctx.ctx.user; + return await model.deleteOneSubRessource({ _id: model._id }, 'paymentInfo.billingInfos', id, ctx); + } +} diff --git a/services/module-payments/functions/orders/__index.ts b/services/module-payments/functions/orders/__index.ts new file mode 100644 index 0000000..ac3046e --- /dev/null +++ b/services/module-payments/functions/orders/__index.ts @@ -0,0 +1,10 @@ +import 'reflect-metadata'; +import OrderEngineResolver from './resolvers/app/app.resolver'; +import OrderAdminEngineQueryResolver from './resolvers/admin/admin.queries'; +import OrderAdminEngineMutationResolver from './resolvers/admin/admin.mutations'; +import OrderBusinessEngineQueryResolver from './resolvers/business/business.queries'; +import OrderBusinessEngineMutationResolver from './resolvers/business/business.mutations'; + +export const OrderEngineResolvers = [OrderEngineResolver]; +export const OrderAdminEngineResolvers = [OrderAdminEngineQueryResolver, OrderAdminEngineMutationResolver]; +export const OrderBusinessEngineResolvers = [OrderBusinessEngineQueryResolver, OrderBusinessEngineMutationResolver]; diff --git a/services/module-payments/functions/orders/carts/__index.ts b/services/module-payments/functions/orders/carts/__index.ts new file mode 100644 index 0000000..f103a45 --- /dev/null +++ b/services/module-payments/functions/orders/carts/__index.ts @@ -0,0 +1,6 @@ +import 'reflect-metadata'; + +import CartEngineResolver from '@services/module-payments/functions/orders/carts/resolvers/cart.resolver'; + +export const CartEngineResolvers = [CartEngineResolver]; +export const CartEngineAdminResolvers = []; diff --git a/services/module-payments/functions/orders/carts/resolvers/cart.resolver.ts b/services/module-payments/functions/orders/carts/resolvers/cart.resolver.ts new file mode 100644 index 0000000..78530f5 --- /dev/null +++ b/services/module-payments/functions/orders/carts/resolvers/cart.resolver.ts @@ -0,0 +1,274 @@ +import _ from 'lodash'; +import { ApolloError } from 'apollo-server-lambda'; + +import { Resolver, Authorized, Ctx, Query } from 'type-graphql'; +import { Mutation, Arg } from 'type-graphql'; + +import { ApolloContext } from '@seed/interfaces/context'; +import { updatePaymentIntent } from '@services/module-payments/helpers/payments.helpers'; +import OrderEngineModel from '../../order.model'; +import PromoModel, { validatePromo } from '@services/module-payments/functions/promos/promo.model'; +import { PaymentIntentInfo } from '@services/module-payments/components/components.payments'; +import { addToCart, getCurrentCartOrder } from '@services/module-payments/functions/orders/services/CartService'; +import { LineItemBaseSchema } from '@src/orders/components/line.schema'; +import { OrderEngineSchema } from '../../schemas/order.schema'; +import { + CartEngineAddCountryCodeSchema, + CartEngineAddPromoSchema, + CartEngineUpdateOrDeleteSchema, + CartEngineUpdateSchema, +} from '../schemas/cart.schema.inputs'; +import { OrderNewInputSchema } from '@src/orders/components/components.cart'; + +@Resolver(OrderEngineSchema) +export default class CartEngineResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + @Query(() => OrderEngineSchema) + @Authorized() + async cartsGetMe(@Ctx() ctx: ApolloContext, @Arg('input', { nullable: true }) input?: OrderNewInputSchema): Promise { + try { + return await getCurrentCartOrder(ctx, input); + } catch (error) { + throw error; + } + } + + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + + @Mutation(() => OrderEngineSchema) + @Authorized() + async cartsAddOneItem(@Arg('input') input: LineItemBaseSchema, @Ctx() ctx: ApolloContext): Promise { + return await addToCart(input, ctx); + } + + @Mutation(() => OrderEngineSchema) + @Authorized() + async cartsUpdateOneItem(@Arg('input') input: CartEngineUpdateSchema, @Ctx() ctx: ApolloContext): Promise { + const account = ctx.ctx.user; + const orderModel = new OrderEngineModel(); + + try { + const currentCartOrder = await getCurrentCartOrder(ctx); + // Check the lines + const lineIndex = _.findIndex(currentCartOrder.lines, function(o) { + if (o.productId == input.productId && _.isEqual(JSON.parse(JSON.stringify(o.options)), JSON.parse(JSON.stringify(o.options)))) + return true; + else return false; + }); + + if (lineIndex === -1) throw new ApolloError('cart.item.notFound', '404', { productId: input.productId }); + + if (input.newQuantity == 0) currentCartOrder.lines.splice(lineIndex, 1); + else { + // Update the quantity + currentCartOrder.lines[lineIndex].quantity = input.newQuantity; + } + + const paymentIntentUpdated: PaymentIntentInfo = await updatePaymentIntent(currentCartOrder.paymentIntent, { + amount: await currentCartOrder.finalPrice(ctx), + currency: currentCartOrder.currency, + account, + }); + + return await orderModel.updateOneCustom({ + query: { _id: currentCartOrder._id }, + updateRequest: { + $set: { + paymentIntent: paymentIntentUpdated, + lines: currentCartOrder.lines, + }, + }, + ctx, + }); + } catch (error) { + throw error; + } + } + + @Mutation(() => OrderEngineSchema) + @Authorized() + async cartsUpdateOrDeleteOneItem(@Arg('input') input: CartEngineUpdateOrDeleteSchema, @Ctx() ctx: ApolloContext): Promise { + const account = ctx.ctx.user; + const orderModel = new OrderEngineModel(); + + const { lineId, newQuantity } = input; + + try { + const currentCartOrder = await getCurrentCartOrder(ctx); + // Check the lines + const lineIndex = _.findIndex(currentCartOrder.lines, { _id: lineId }); + + if (lineIndex === -1) throw new ApolloError('cart.line.notFound', '404', { lineId: input.lineId }); + + if (newQuantity == 0) currentCartOrder.lines.splice(lineIndex, 1); + else { + // Update the quantity + currentCartOrder.lines[lineIndex].quantity = newQuantity; + } + + const paymentIntentUpdated: PaymentIntentInfo = await updatePaymentIntent(currentCartOrder.paymentIntent, { + amount: await currentCartOrder.finalPrice(ctx), + currency: currentCartOrder.currency, + account, + }); + + return await orderModel.updateOneCustom({ + query: { _id: currentCartOrder._id }, + updateRequest: { + $set: { + paymentIntent: paymentIntentUpdated, + lines: currentCartOrder.lines, + }, + }, + ctx, + }); + } catch (error) { + throw error; + } + } + + @Mutation(() => OrderEngineSchema) + @Authorized() + async cartsEmpty(@Ctx() ctx: ApolloContext): Promise { + const orderModel = new OrderEngineModel(); + + try { + const currentCartOrder = await getCurrentCartOrder(ctx); + return await orderModel.updateOneCustom({ + query: { _id: currentCartOrder._id }, + updateRequest: { + $set: { + lines: [], + }, + $unset: { + promoId: '', + countryCode: '', + locale: '', + }, + }, + ctx, + }); + } catch (error) { + throw error; + } + } + + @Mutation(() => OrderEngineSchema) + @Authorized() + async cartAddCountryCode(@Arg('input') input: CartEngineAddCountryCodeSchema, @Ctx() ctx: ApolloContext): Promise { + const orderModel = new OrderEngineModel(); + + try { + const currentCartOrder = await getCurrentCartOrder(ctx); + if (!currentCartOrder) throw new ApolloError('cart.notFound', '404'); + + const lines = currentCartOrder.lines; + lines.map((line) => { + if (line.localeInfo) return (line.localeInfo.countryCode = input.countryCode); + return (line.localeInfo = { countryCode: input.countryCode }); + }); + + return await orderModel.updateOneCustom({ + query: { _id: currentCartOrder._id }, + updateRequest: { + $set: { countryCode: input.countryCode, lines }, + }, + ctx, + }); + } catch (error) { + throw error; + } + } + + @Mutation(() => OrderEngineSchema) + @Authorized() + async cartAddPromo(@Arg('input') input: CartEngineAddPromoSchema, @Ctx() ctx: ApolloContext): Promise { + const orderModel = new OrderEngineModel(); + const promoModel = new PromoModel(); + + try { + const currentCartOrder = await getCurrentCartOrder(ctx); + + if (!currentCartOrder) throw new ApolloError('cart.notFound', '404'); + + await promoModel.getOne({ code: input.promoCode }, ctx); + await validatePromo(input.promoCode, ctx); + + const lines = currentCartOrder.lines; + lines.map((line) => { + return (line.promoId = promoModel._id); + }); + + const paymentIntentUpdated: PaymentIntentInfo = await updatePaymentIntent(currentCartOrder.paymentIntent, { + amount: await currentCartOrder.finalPrice(ctx), + currency: currentCartOrder.currency, + account: ctx.ctx.user, + }); + + return await orderModel.updateOneCustom({ + query: { _id: currentCartOrder._id }, + updateRequest: { + // finalPrice: currentCart.finalPrice, + $set: { promoId: promoModel._id, lines, paymentIntent: paymentIntentUpdated }, + }, + ctx, + }); + } catch (error) { + throw error; + } + } + + @Mutation(() => OrderEngineSchema) + @Authorized() + async cartRemovePromo(@Ctx() ctx: ApolloContext): Promise { + const orderModel = new OrderEngineModel(); + + try { + const currentCartOrder = await getCurrentCartOrder(ctx); + + if (!currentCartOrder) throw new ApolloError('cart.notFound', '404'); + + const lines = currentCartOrder.lines; + lines.forEach((line) => { + delete line.promoId; + }); + + const paymentIntentUpdated: PaymentIntentInfo = await updatePaymentIntent(currentCartOrder.paymentIntent, { + amount: await currentCartOrder.finalPrice(ctx), + currency: currentCartOrder.currency, + account: ctx.ctx.user, + }); + + return await orderModel.updateOneCustom({ + query: { _id: currentCartOrder._id }, + updateRequest: { + $unset: { + promoId: '', + }, + $set: { + lines, + paymentIntent: paymentIntentUpdated, + }, + }, + ctx, + }); + } catch (error) { + throw error; + } + } +} diff --git a/services/module-payments/functions/orders/carts/schemas/cart.schema.inputs.ts b/services/module-payments/functions/orders/carts/schemas/cart.schema.inputs.ts new file mode 100644 index 0000000..42e08dc --- /dev/null +++ b/services/module-payments/functions/orders/carts/schemas/cart.schema.inputs.ts @@ -0,0 +1,30 @@ +import { AvailableTranslation } from '@src/__components/components'; +import { InputType, Field } from 'type-graphql'; + +@InputType() +export class CartEngineUpdateSchema { + @Field() + productId: string; + @Field() + newQuantity: number; +} + +@InputType() +export class CartEngineUpdateOrDeleteSchema { + @Field() + lineId: string; + @Field() + newQuantity: number; +} + +@InputType() +export class CartEngineAddPromoSchema { + @Field() + promoCode: string; +} + +@InputType() +export class CartEngineAddCountryCodeSchema { + @Field() + countryCode: string; +} diff --git a/services/module-payments/functions/orders/carts/schemas/line.schema.ts b/services/module-payments/functions/orders/carts/schemas/line.schema.ts new file mode 100644 index 0000000..f909952 --- /dev/null +++ b/services/module-payments/functions/orders/carts/schemas/line.schema.ts @@ -0,0 +1,134 @@ +import { ObjectType, Field, InputType } from 'type-graphql'; +import { ProductRessourceEnum } from '@src/__components/components'; +import { GraphQLJSONObject } from 'graphql-type-json'; +import { AddOnComponent, LocaleInfo } from '@services/module-payments/components/components.orders'; + +export interface LineProductInterface { + _id: string; + + productRessource: ProductRessourceEnum; + productId: string; + + title: string; + + price: number; + salesPrice?: number; + + quantity: number; + + vatClassId?: string; + + // vatPrice(): number; + + finalPrice(): number; + + // linePrice(): number; + + organisationId?: string; + parentId?: string; + + addOns?: AddOnComponent[]; + + localeInfo?: LocaleInfo; +} + +export interface LineProductFunctionInterface { + validateCartOption?(data: any): any; + getLinePrice?(data: any): any; +} + +@ObjectType() +export class LineProductClass implements LineProductInterface { + // linePrice(): number { + // throw new Error('Method not implemented.'); + // } + @Field() + _id: string; + + @Field(() => ProductRessourceEnum) + productRessource: ProductRessourceEnum; + + @Field() + productId: string; + + @Field() + title: string; + + @Field() + price: number; + @Field() + salesPrice?: number; + + @Field() + quantity: number; + + @Field() + vatClassId?: string; + + @Field({ nullable: true }) + parentId?: string; + + @Field(() => [AddOnComponent], { nullable: true }) + addOns?: AddOnComponent[]; + + @Field(() => LocaleInfo, { nullable: true }) + localeInfo?: LocaleInfo; + + @Field({ nullable: true }) + promoId?: string; + + + // @Field() + // vatPrice(): number { + // throw 'Implements on parent'; + // } + + validateCartOption?(): any; + organisationId?: string; + + @Field({ nullable: true }) + finalPrice(): number { + if (this.salesPrice) return this.salesPrice; + else return this.price; + } +} + +@ObjectType() +@InputType('LineProductBaseInput') +export class LineProductBase { + @Field(() => ProductRessourceEnum) + productRessource: ProductRessourceEnum; + + @Field() + productId: string; + + @Field() + quantity: number; + + @Field({ nullable: true }) + parentId?: string; + + @Field(() => GraphQLJSONObject, { nullable: true }) + options?: any; + + @Field(() => [AddOnComponent], { nullable: true }) + addOns?: AddOnComponent[]; +} + +@ObjectType() +@InputType('LineProductBaseSuppInput') +export class LineProductBaseSupp extends LineProductBase { + @Field() + parentId: string; +} + +export interface LineProductBaseInterface { + getLineProduct(input: { quantity: number }): LineProductInterface; + getLineProductFunctions(input: any): LineProductFunctionInterface; +} + +@InputType('LineItemGenericSchemaInput') +@ObjectType() +export abstract class LineItemGenericSchema extends LineProductClass { + options?: any; +} diff --git a/services/module-payments/functions/orders/helpers/OrderHelper.ts b/services/module-payments/functions/orders/helpers/OrderHelper.ts new file mode 100644 index 0000000..142acdb --- /dev/null +++ b/services/module-payments/functions/orders/helpers/OrderHelper.ts @@ -0,0 +1,35 @@ +import { ApolloContext } from '@lib/seed/interfaces/context'; +import { OrderEngineDBInterfaceSchema } from '../schemas/order.schema'; + +export async function getTotalPrice(order: OrderEngineDBInterfaceSchema, ctx: ApolloContext): Promise { + let price = 0; + if (order.lines && order.lines.length > 0) { + for (let index = 0; index < order.lines.length; index++) { + const line = order.lines[index]; + price += (await line.getLinePrice(ctx)) + (await line.getLineVatPrice(ctx)); + } + } + return price; +} + +export async function getVatPrice(order: OrderEngineDBInterfaceSchema, ctx: ApolloContext): Promise { + let price = 0; + if (order.lines && order.lines.length > 0) { + for (let index = 0; index < order.lines.length; index++) { + const line = order.lines[index]; + price += await line.getLineVatPrice(ctx); + } + } + return price; +} + +export async function getSubTotalPrice(order: OrderEngineDBInterfaceSchema, ctx: ApolloContext): Promise { + let price = 0; + if (order.lines && order.lines.length > 0) { + for (let index = 0; index < order.lines.length; index++) { + const line = order.lines[index]; + price += await line.getLinePrice(ctx); + } + } + return price; +} diff --git a/services/module-payments/functions/orders/order.model.ts b/services/module-payments/functions/orders/order.model.ts new file mode 100644 index 0000000..e18e4f8 --- /dev/null +++ b/services/module-payments/functions/orders/order.model.ts @@ -0,0 +1,60 @@ +import { ObjectType, Field, ID, ArgsType, InputType } from 'type-graphql'; + +import { Permission } from '@seed/interfaces/permission'; + +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { OrderEngineDBInterfaceSchema, OrderEngineDBSchema, OrderEngineSchema } from './schemas/order.schema'; +import { EngineModel } from '@lib/seed/engine/EngineModel'; +import { plainToClass } from 'class-transformer'; +import { IEngineSchema } from '@lib/seed/engine/EngineSchema'; +import { OrderArgs } from './schemas/order.schema.input'; +import { CustomSearchEngine } from '@lib/seed/services/database/DBRequestService'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.admin], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; +@ObjectType() +export default class OrderEngineModel extends EngineModel { + public constructor(input?: OrderEngineDBInterfaceSchema & Partial) { + const dataInit = plainToClass(OrderEngineDBSchema, input || {}); + super({ + // ...init, + collectionName: 'shop.orders', + permissions: permissions, + dataInit, + }); + } + + plainToClass(plain: any): OrderEngineSchema | OrderEngineSchema[] { + return plainToClass(OrderEngineSchema, plain); + } + + searchEngine(): CustomSearchEngine { + const sEngine: CustomSearchEngine = { + orderStatus: { + operation: '$eq', + }, + paymentStatus: { + operation: '$eq', + }, + search: { + operation: 'freeInput', + freeInputConfig: { + searchOptions: ['billingInfo.company', 'billingInfo.email'], + }, + }, + }; + return sEngine; + } + + async afterPaymentSuccess(orderData: OrderEngineSchema): Promise { + throw 'To Implement on parent'; + } + + async afterPaymentFailed(orderData: OrderEngineSchema): Promise { + throw 'To Implement on parent'; + } +} diff --git a/services/module-payments/functions/orders/resolvers/admin/admin.mutations.ts b/services/module-payments/functions/orders/resolvers/admin/admin.mutations.ts new file mode 100644 index 0000000..c9bab71 --- /dev/null +++ b/services/module-payments/functions/orders/resolvers/admin/admin.mutations.ts @@ -0,0 +1,116 @@ +import { Arg, Args, Ctx, Mutation, Resolver, Authorized } from 'type-graphql'; +import { ApolloContext } from '@lib/seed/interfaces/context'; +import OrderEngineModel from '@services/module-payments/functions/orders/order.model'; +import { OrderEngineSchema } from '../../schemas/order.schema'; +import { createBasePublicResolver } from '@seed/graphql/baseResolvers/BasePublicResolver'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { ApolloError } from 'apollo-server-lambda'; + +import StripeService from '@services/module-payments/services/stripe/StripeService'; +import { OrderStatusEnum } from '@services/module-payments/components/components.orders'; +import { PaymentStatusEnum } from '@services/module-payments/components/components.payments'; +import { AsyncHooksService } from '@lib/__hooks'; + +@Resolver(OrderEngineModel) +export default class OrderAdminEngineMutationResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + @Mutation(() => OrderEngineSchema) + @Authorized(AccountTypeEnum.admin) + async ordersMarkOneAsPaid(@Arg('id') orderId: string, @Ctx() ctx: ApolloContext): Promise { + const model = new OrderEngineModel(); + const currentOrder = await model.getOne({ query: { _id: orderId }, ctx }); + + if (currentOrder.paymentStatus == PaymentStatusEnum.paid) throw new ApolloError('orders.payment.alreadypaid', '400'); + if (currentOrder.paymentStatus == PaymentStatusEnum.free) throw new ApolloError('orders.payment.freeItem', '400'); + + const result = await model.updateOne({ + query: { _id: currentOrder._id }, + newData: { + paymentStatus: PaymentStatusEnum.paid, + }, + ctx, + }); + await new AsyncHooksService().afterOrdersMarkOneAsPaid(result, ctx); + return result; + } + + @Mutation(() => OrderEngineSchema) + @Authorized(AccountTypeEnum.admin) + async ordersMarkOneAsUnPaid(@Arg('id') orderId: string, @Ctx() ctx: ApolloContext): Promise { + const model = new OrderEngineModel(); + const currentOrder = await model.getOne({ query: { _id: orderId }, ctx }); + + if (currentOrder.paymentStatus == PaymentStatusEnum.pending) throw new ApolloError('orders.payment.alreadyUnPaid', '400'); + if (currentOrder.paymentStatus == PaymentStatusEnum.free) throw new ApolloError('orders.payment.freeItem', '400'); + + const result = await model.updateOne({ + query: { _id: currentOrder._id }, + newData: { + paymentStatus: PaymentStatusEnum.pending, + }, + ctx, + }); + await new AsyncHooksService().afterOrdersMarkOneAsUnPaid(result, ctx); + return result; + } + + @Mutation(() => OrderEngineSchema) + @Authorized(AccountTypeEnum.admin) + async ordersCancelOne(@Arg('id') orderId: string, @Ctx() ctx: ApolloContext): Promise { + const model = new OrderEngineModel(); + const currentOrder = await model.getOne({ query: { _id: orderId }, ctx }); + + if (currentOrder.orderStatus == OrderStatusEnum.canceled) throw new ApolloError('orders.alreadyCancel', '400'); + if (currentOrder.orderStatus == OrderStatusEnum.refund) throw new ApolloError('orders.alreadyRefund', '400'); + + const result = await model.updateOne({ + query: { _id: currentOrder._id }, + newData: { + orderStatus: OrderStatusEnum.canceled, + }, + ctx, + }); + await new AsyncHooksService().afterOrderCancel(result, ctx); + return result; + } + + @Mutation(() => OrderEngineSchema) + @Authorized(AccountTypeEnum.admin) + async ordersReimburseOne(@Arg('orderId') orderId: string, @Ctx() ctx: ApolloContext): Promise { + const model = new OrderEngineModel(); + const currentOrder = await model.getOne({ query: { _id: orderId }, ctx }); + if (currentOrder.orderStatus != OrderStatusEnum.canceled) throw new ApolloError('orders.reimburse.notCanceled', '400'); + try { + if (currentOrder.paymentInfo?.transactionId) { + await StripeService.getInstance().reimburse(currentOrder.paymentInfo.transactionId); + const result = await model.updateOne({ + query: { _id: currentOrder._id }, + newData: { + orderStatus: OrderStatusEnum.refund, + }, + ctx, + }); + await new AsyncHooksService().afterOrdersReimbursed(result, ctx); + return result; + } + throw new ApolloError('orders.reimburse.noTransactionFound', '400'); + } catch (error) { + throw error; + } + } +} diff --git a/services/module-payments/functions/orders/resolvers/admin/admin.queries.ts b/services/module-payments/functions/orders/resolvers/admin/admin.queries.ts new file mode 100644 index 0000000..4ae1526 --- /dev/null +++ b/services/module-payments/functions/orders/resolvers/admin/admin.queries.ts @@ -0,0 +1,46 @@ +import { Arg, Args, Ctx, Query, Resolver } from 'type-graphql'; + +import OrderModel from '@services/module-payments/functions/orders/order.model'; +import { ApolloContext } from '@lib/seed/interfaces/context'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { OrderEngineSchema } from '../../schemas/order.schema'; +import { OrderArgs } from '../../schemas/order.schema.input'; +import { EngineMiddleware } from '@lib/seed/graphql/MiddlewareV2'; + +@Resolver(OrderModel) +export default class OrderAdminEngineQueryResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + @Query(() => OrderEngineSchema) + @EngineMiddleware({ + authorization: [AccountTypeEnum.admin], + }) + async adminsOrdersGetOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + return new OrderModel().getOneGeneric(id, ctx); + } + + @Query(() => [OrderEngineSchema]) + @EngineMiddleware({ + authorization: [AccountTypeEnum.admin], + }) + async adminsOrdersGetMany(@Args() args: OrderArgs, @Ctx() ctx: ApolloContext): Promise { + const results = await new OrderModel().getManyGenericWithArgs(args, ctx); + return results; + } + + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ +} diff --git a/services/module-payments/functions/orders/resolvers/app/app.resolver.ts b/services/module-payments/functions/orders/resolvers/app/app.resolver.ts new file mode 100644 index 0000000..652a16e --- /dev/null +++ b/services/module-payments/functions/orders/resolvers/app/app.resolver.ts @@ -0,0 +1,86 @@ +import { Resolver, Authorized, Query, Ctx, Arg, Args, Mutation } from 'type-graphql'; +import { ApolloContext } from '@seed/interfaces/context'; + +import OrderModel from '@services/module-payments/functions/orders/order.model'; +import { OrderEngineSchema } from '../../schemas/order.schema'; +import { OrderArgs } from '../../schemas/order.schema.input'; +import { getCurrentCartOrder } from '../../services/CartService'; +import { newError } from '@seed/helpers/Error'; +import { localCheckout } from '../../services/OrderService'; + +@Resolver(OrderModel) +export default class OrderEngineResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + @Query(() => OrderEngineSchema) + @Authorized() + async myCurrentOrder(@Ctx() ctx: ApolloContext): Promise { + const cartOrderData = await getCurrentCartOrder(ctx); + if (!cartOrderData) throw newError(6003); + + return cartOrderData; + } + + @Query(() => OrderEngineSchema) + @Authorized() + async myOrdersGetOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const account = ctx.ctx.user; + + return new OrderModel().getOne({ query: { _id: id, accountId: account._id } }); + } + + @Query(() => [OrderEngineSchema]) + @Authorized() + async myOrdersGetMany(@Args() arg: OrderArgs, @Ctx() ctx: ApolloContext): Promise { + const account = ctx.ctx.user; + const args = { ...arg, accountId: account._id }; + + return new OrderModel().getManyGenericWithArgs(args, ctx); + } + + @Query(() => Number) + @Authorized() + async myOrdersGetManyCount(@Args() arg: OrderArgs, @Ctx() ctx: ApolloContext): Promise { + const account = ctx.ctx.user; + const args = { ...arg, accountId: account._id }; + + return new OrderModel().getCountGeneric(args, ctx); + } + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + + @Mutation(() => OrderEngineSchema) + @Authorized() + async removeCurrentOrder(@Ctx() ctx: ApolloContext): Promise { + try { + const account = ctx.ctx.user; + return new OrderModel().deleteOne({ query: { accountId: account._id }, ctx }); + } catch (error) { + throw error; + } + } + + @Mutation(() => OrderEngineSchema) + async validateOrder(@Arg('id') id: string): Promise { + try { + const order = await new OrderModel().getOne({ query: { _id: id } }); + await localCheckout(order); + return order; + } catch (error) { + throw error; + } + } +} diff --git a/services/module-payments/functions/orders/resolvers/app/checkout.resolver.ts b/services/module-payments/functions/orders/resolvers/app/checkout.resolver.ts new file mode 100644 index 0000000..447197e --- /dev/null +++ b/services/module-payments/functions/orders/resolvers/app/checkout.resolver.ts @@ -0,0 +1,54 @@ +import { Resolver, Authorized, Ctx } from 'type-graphql'; +import { Mutation, Arg } from 'type-graphql'; +import { ApolloContext } from '@seed/interfaces/context'; + +// From Module +import { OrderEngineSchema } from '../../schemas/order.schema'; +import { CheckoutEngineInput } from '../../schemas/order.schema.input'; + +// From SRC +import OrderModel from '@src/orders/order.model'; +import { createCheckout, finaliseCheckoutWithOrder } from '../../services/OrderService'; +import { PaymentProvider } from '@services/module-payments/components/components.payments'; +import { CheckoutInput } from '@src/orders/components/checkout.schema'; + +@Resolver(OrderModel) +export default class CheckoutEngineResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + + @Mutation(() => OrderEngineSchema) + @Authorized() + async finaliseCheckoutWithStripe(@Arg('input') input: CheckoutInput, @Ctx() ctx: ApolloContext): Promise { + try { + const { billingInfo, contactInfo, ...rest } = input; + return await finaliseCheckoutWithOrder({ + type: 'default', + payment: PaymentProvider.stripe, + contactInfo: { + billingInfo, + contactInfo, + }, + input: rest, + ctx, + }); + } catch (error) { + throw error; + } + } +} diff --git a/services/module-payments/functions/orders/resolvers/business/business.input.ts b/services/module-payments/functions/orders/resolvers/business/business.input.ts new file mode 100644 index 0000000..454d827 --- /dev/null +++ b/services/module-payments/functions/orders/resolvers/business/business.input.ts @@ -0,0 +1,7 @@ +import { InputType, Field } from 'type-graphql'; + +@InputType() +export class ChangeStatusInput { + @Field({ nullable: true }) + customMessage?: string; +} diff --git a/services/module-payments/functions/orders/resolvers/business/business.mutations.ts b/services/module-payments/functions/orders/resolvers/business/business.mutations.ts new file mode 100644 index 0000000..b2a7b41 --- /dev/null +++ b/services/module-payments/functions/orders/resolvers/business/business.mutations.ts @@ -0,0 +1,58 @@ +import { Arg, Args, Ctx, Mutation, Query, Resolver } from 'type-graphql'; +import _ from 'lodash'; + +import OrderModel from '@services/module-payments/functions/orders/order.model'; +import { ApolloContext } from '@lib/seed/interfaces/context'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { ProviderSchema } from '../../schemas/provider.schema'; +import { EngineMiddleware } from '@lib/seed/graphql/MiddlewareV2'; +import { updateBusinessOrderStatus } from './business.service'; +import { ChangeStatusInput } from './business.input'; +import { OrderBusinessStatusEnum } from '@services/module-payments/components/components.orders'; + +@Resolver(OrderModel) +export default class OrderBusinessEngineMutationResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + @Mutation(() => ProviderSchema) + @EngineMiddleware({ + authorization: [AccountTypeEnum.organisationOwner, AccountTypeEnum.storeManager], + checkOrganisation: true, + }) + async businessOrdersMarkAsShipped( + @Arg('id') id: string, + @Arg('input') input: ChangeStatusInput, + @Ctx() ctx: ApolloContext, + ): Promise { + return await updateBusinessOrderStatus(id, OrderBusinessStatusEnum.shipped, ctx); + } + + @Mutation(() => ProviderSchema) + @EngineMiddleware({ + authorization: [AccountTypeEnum.organisationOwner, AccountTypeEnum.storeManager], + checkOrganisation: true, + }) + async businessOrdersMarkAsCompleted( + @Arg('id') id: string, + @Arg('input') input: ChangeStatusInput, + @Ctx() ctx: ApolloContext, + ): Promise { + return await updateBusinessOrderStatus(id, OrderBusinessStatusEnum.completed, ctx); + } + + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ +} diff --git a/services/module-payments/functions/orders/resolvers/business/business.queries.ts b/services/module-payments/functions/orders/resolvers/business/business.queries.ts new file mode 100644 index 0000000..4dba85d --- /dev/null +++ b/services/module-payments/functions/orders/resolvers/business/business.queries.ts @@ -0,0 +1,67 @@ +import { Arg, Args, Ctx, Query, Resolver } from 'type-graphql'; + +import OrderModel from '@services/module-payments/functions/orders/order.model'; +import { ApolloContext } from '@lib/seed/interfaces/context'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { ProviderSchema } from '../../schemas/provider.schema'; +import _ from 'lodash'; +import { OrderBusinessArgs } from '../../schemas/order.schema.input'; +import { EngineMiddleware } from '@lib/seed/graphql/MiddlewareV2'; +import { OrderStatusEnum } from '@services/module-payments/components/components.orders'; +import { PaymentStatusEnum } from '@services/module-payments/components/components.payments'; + +@Resolver(OrderModel) +export default class OrderBusinessEngineQueryResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + @Query(() => ProviderSchema) + @EngineMiddleware({ + authorization: [AccountTypeEnum.organisationOwner, AccountTypeEnum.storeManager], + checkOrganisation: true, + }) + async businessOrdersGetOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const organisationId = ctx.ctx.organisationId; + const order = await new OrderModel().getOne({ query: { 'providerOrderItems.organisationId': organisationId, 'providerOrderItems._id': id } }); + + return _.find(order.providerOrderItems, { _id: id }); + } + + @Query(() => [ProviderSchema], { nullable: 'itemsAndList' }) + @EngineMiddleware({ + authorization: [AccountTypeEnum.organisationOwner, AccountTypeEnum.storeManager], + checkOrganisation: true, + }) + async businessOrdersGetMany(@Args() args: OrderBusinessArgs, @Ctx() ctx: ApolloContext): Promise { + // eslint-disable-next-line @typescript-eslint/no-non-null-assertion + const organisationId = ctx.ctx.organisationId!; + const { pagination, ...arg } = args; + + // Get only those where orderStatus and paymentStatus is correct + + const filters: any = { + 'providerOrderItems.organisationId': organisationId, + orderStatus: { $in: [OrderStatusEnum.processing, OrderStatusEnum.shipped, OrderStatusEnum.completed] }, + paymentStatus: { $in: [PaymentStatusEnum.paid] }, + }; + if (arg.orderStatus) filters['providerOrderItems.orderStatus'] = { $in: arg.orderStatus }; + + const orders = await new OrderModel().getMany({ query: filters, pagination }); + const results: ProviderSchema[] = []; + + orders.forEach((order) => { + if (order.providerOrderItems) { + const index = _.findIndex(order.providerOrderItems, { organisationId: organisationId }); + if (index !== -1) results.push(order.providerOrderItems[index]); + } + }); + + return results; + } +} diff --git a/services/module-payments/functions/orders/resolvers/business/business.service.ts b/services/module-payments/functions/orders/resolvers/business/business.service.ts new file mode 100644 index 0000000..17470e7 --- /dev/null +++ b/services/module-payments/functions/orders/resolvers/business/business.service.ts @@ -0,0 +1,23 @@ +import { ApolloContext } from '@lib/seed/interfaces/context'; +import { OrderBusinessStatusEnum } from '@services/module-payments/components/components.orders'; +import { ApolloError } from 'apollo-server-lambda'; +import _ from 'lodash'; +import OrderEngineModel from '../../order.model'; +import { ProviderSchema } from '../../schemas/provider.schema'; + +export const updateBusinessOrderStatus = async (id: string, newStatus: OrderBusinessStatusEnum, ctx: ApolloContext): Promise => { + const organisationId = ctx.ctx.organisationId; + const orderModel = new OrderEngineModel(); + const order = await orderModel.getOne({ query: { 'providerOrderItems.organisationId': organisationId, 'providerOrderItems._id': id } }); + + const providerOrderItem = _.find(order.providerOrderItems, { _id: id }); + if (!providerOrderItem) throw new ApolloError('notfound', '400'); + + providerOrderItem.orderStatus = newStatus; + await (await orderModel.db()).updateOne( + { 'providerOrderItems.organisationId': organisationId, 'providerOrderItems._id': id }, + { $set: { 'providerOrderItems.$': providerOrderItem } }, + ); + + return providerOrderItem; +}; diff --git a/services/module-payments/functions/orders/resolvers/field.resolver.ts b/services/module-payments/functions/orders/resolvers/field.resolver.ts new file mode 100644 index 0000000..43d9655 --- /dev/null +++ b/services/module-payments/functions/orders/resolvers/field.resolver.ts @@ -0,0 +1,9 @@ +import { Resolver } from 'type-graphql'; +import { OrderEngineSchema } from '../schemas/order.schema'; +import { ProviderSchema } from '../schemas/provider.schema'; + +@Resolver(() => OrderEngineSchema) +export class OrderFieldResolver {} + +@Resolver(() => ProviderSchema) +export class ProviderItemFieldResolver {} diff --git a/services/module-payments/functions/orders/schemas/order.schema.input.ts b/services/module-payments/functions/orders/schemas/order.schema.input.ts new file mode 100644 index 0000000..000190a --- /dev/null +++ b/services/module-payments/functions/orders/schemas/order.schema.input.ts @@ -0,0 +1,49 @@ +import { GetArgs } from '@lib/seed/graphql/Request'; +import { LocaleInfo, OrderStatusEnum } from '@services/module-payments/components/components.orders'; +import { PaymentStatusEnum } from '@services/module-payments/components/components.payments'; + +import { AvailableTranslation, BuyerInfoComponent } from '@src/__components/components'; +import { ArgsType, Field, InputType, ObjectType } from 'type-graphql'; + +@ArgsType() +export class OrderArgs { + @Field(() => OrderStatusEnum, { nullable: true }) + orderStatus?: OrderStatusEnum; + + @Field(() => PaymentStatusEnum, { nullable: true }) + paymentStatus?: PaymentStatusEnum; + + @Field(() => GetArgs, { nullable: true }) + pagination?: GetArgs; + + @Field({ nullable: true }) + search?: string; +} + +@ArgsType() +export class OrderBusinessArgs { + @Field(() => [OrderStatusEnum], { nullable: true }) + orderStatus?: OrderStatusEnum[]; + + @Field(() => GetArgs, { nullable: true }) + pagination?: GetArgs; +} + +@ObjectType() +@InputType('CheckoutEngineInputI') +export class CheckoutEngineInput { + @Field(() => BuyerInfoComponent) + contactInfo: BuyerInfoComponent; + + @Field(() => BuyerInfoComponent) + billingInfo: BuyerInfoComponent; + + @Field({ nullable: true }) + token?: string; +} + +@InputType() +export class OrderEngineNewSchema { + @Field(() => LocaleInfo, { nullable: true }) + localeInfo?: AvailableTranslation; +} diff --git a/services/module-payments/functions/orders/schemas/order.schema.ts b/services/module-payments/functions/orders/schemas/order.schema.ts new file mode 100644 index 0000000..da9ce45 --- /dev/null +++ b/services/module-payments/functions/orders/schemas/order.schema.ts @@ -0,0 +1,140 @@ +import { ObjectType, Field, InputType, ID, Ctx } from 'type-graphql'; + +import { IEngineSchema, MetaBy, MetaPermissions } from '@lib/seed/engine/EngineSchema'; +import { AvailableTranslation, BuyerInfoComponent, TranslatableComponent } from '@src/__components/components'; +import { ProviderSchema } from './provider.schema'; +import { Type } from 'class-transformer'; +import { LocaleInfo, OrderStatusEnum } from '@services/module-payments/components/components.orders'; +import { + PaymentIntentInfo, + PaymentStatusEnum, + PaymentInfo, + InvoiceInfo, + CurrencyEnum, +} from '@services/module-payments/components/components.payments'; +import { ApolloContext } from '@seed/interfaces/context'; +import { getTotalPrice, getSubTotalPrice, getVatPrice } from '../helpers/OrderHelper'; +import { PromoType } from '@services/module-payments/functions/promos/component'; +import PromoModel from '@services/module-payments/functions/promos/promo.model'; +import { LineItemSchema } from '@src/orders/components/line.schema'; +import { EnginePathComponent } from '@seed/interfaces/components'; + +@InputType() +@ObjectType() +export class OrderEngineBaseSchema { + /* + LINES + */ + + @Field(() => [LineItemSchema]) + @Type(() => LineItemSchema) + lines: LineItemSchema[]; + + /* + CURRENCY & LOCALE + */ + + @Field(() => LocaleInfo) + localeInfo: LocaleInfo; + + @Field(() => CurrencyEnum) + currency: CurrencyEnum; + + @Field(() => Boolean) + vatExempt = false; + + /* + CONTACTS + */ + + @Field() + accountId: string; + + @Field(() => BuyerInfoComponent, { nullable: true }) + contactInfo?: BuyerInfoComponent; + + @Field(() => BuyerInfoComponent, { nullable: true }) + billingInfo?: BuyerInfoComponent; + + /* + PROMOTIONS + */ + + @Field(() => ID, { nullable: true }) + promoId?: string; + + /* + INVOICING + */ + + @Field(() => InvoiceInfo, { nullable: true }) + @Type(() => InvoiceInfo) + invoiceInfo?: InvoiceInfo; + + /* + PROVIDERS + */ + @Field(() => [ProviderSchema], { nullable: true }) + @Type(() => ProviderSchema) + providerOrderItems?: ProviderSchema[]; + + @Field(() => TranslatableComponent, { nullable: true }) + orderName?: TranslatableComponent; +} + +@ObjectType() +export class OrderEngineDBInterfaceSchema extends OrderEngineBaseSchema { + @Field(() => OrderStatusEnum) + orderStatus: OrderStatusEnum; + + @Field(() => PaymentIntentInfo) + paymentIntent: PaymentIntentInfo; + + @Field(() => PaymentStatusEnum, { nullable: true }) + paymentStatus?: PaymentStatusEnum; + @Field(() => PaymentInfo, { nullable: true }) + paymentInfo?: PaymentInfo; + + inputData?: any; +} + +@ObjectType() +export class OrderEngineDBSchema extends OrderEngineDBInterfaceSchema { + @Field(() => Number) + async subTotalPrice(@Ctx() ctx: ApolloContext): Promise { + return await getSubTotalPrice(this, ctx); + } + + @Field(() => Number) + async vatPrice(@Ctx() ctx: ApolloContext): Promise { + return await getVatPrice(this, ctx); + } + + @Field(() => Number) + async finalPrice(@Ctx() ctx: ApolloContext): Promise { + if (this.vatExempt) return getSubTotalPrice(this, ctx); + return await getTotalPrice(this, ctx); + } + + @Field(() => Number) + async totalPrice(@Ctx() ctx: ApolloContext): Promise { + return this.finalPrice(ctx) + } + + @Field(() => PromoModel, { nullable: true }) + async promo(@Ctx() ctx: ApolloContext): Promise { + if (this.promoId) return await ctx.ctx.loaders.promoLoader.load(this.promoId); + return; + } +} + +@ObjectType({ implements: IEngineSchema }) +export class OrderEngineSchema extends OrderEngineDBSchema implements IEngineSchema { + _id: string; + organisationId?: string | undefined; + by?: MetaBy | undefined; + paths?: EnginePathComponent[]; + permissions: MetaPermissions; + createdAt: Date; + updatedAt: Date; +} diff --git a/services/module-payments/functions/orders/schemas/provider.schema.ts b/services/module-payments/functions/orders/schemas/provider.schema.ts new file mode 100644 index 0000000..a4cb512 --- /dev/null +++ b/services/module-payments/functions/orders/schemas/provider.schema.ts @@ -0,0 +1,19 @@ +import { OrderBusinessStatusEnum } from '@services/module-payments/components/components.orders'; +import { LineItemSchema } from '@src/orders/components/line.schema'; +import { ObjectType, Field, InputType } from 'type-graphql'; + +@InputType() +@ObjectType() +export class ProviderSchema { + @Field() + _id: string; + + @Field() + organisationId: string; + + @Field(() => OrderBusinessStatusEnum) + orderStatus: OrderBusinessStatusEnum; + + @Field(() => [LineItemSchema]) + lines: LineItemSchema[]; +} diff --git a/services/module-payments/functions/orders/services/CartService.ts b/services/module-payments/functions/orders/services/CartService.ts new file mode 100644 index 0000000..4486fac --- /dev/null +++ b/services/module-payments/functions/orders/services/CartService.ts @@ -0,0 +1,113 @@ +import { ApolloContext } from '@lib/seed/interfaces/context'; +import { OrderStatusEnum } from '@services/module-payments/components/components.orders'; +import { PaymentIntentInfo } from '@services/module-payments/components/components.payments'; +import { updatePaymentIntent } from '@services/module-payments/helpers/payments.helpers'; +import { OrderNewInputSchema } from '@src/orders/components/components.cart'; +import { LineItemBaseSchema, LineItemDBSchema } from '@src/orders/components/line.schema'; +import OrderModel from '@src/orders/order.model'; +import { ApolloError } from 'apollo-server-lambda'; +import { plainToClass } from 'class-transformer'; +import _ from 'lodash'; +import { LineProductBase } from '../carts/schemas/line.schema'; +import OrderEngineModel from '../order.model'; +import { OrderEngineSchema } from '../schemas/order.schema'; +import { createDraftOrder } from './OrderService'; + +import { v4 as uuid } from 'uuid'; + +export const getCurrentCartOrder = async (ctx: ApolloContext, input?: OrderNewInputSchema): Promise => { + const account = ctx.ctx.user; + const orderModel = new OrderModel(); + + try { + try { + return await orderModel.getOne({ query: { orderStatus: OrderStatusEnum.draft, accountId: account._id } }); + } catch (error) { + const co: any = { + ctx, + }; + if (input?.localeInfo) { + co.localeInfo = input.localeInfo; + delete input.localeInfo; + } + co.input = input; + + return await createDraftOrder(co); + } + } catch (error) { + throw error; + } +}; + +export const addToCart = async (input: T, ctx: ApolloContext): Promise => { + const orderModel = new OrderEngineModel(); + const account = ctx.ctx.user; + + const { productRessource, productId, quantity, parentId, options } = input; + + try { + const currentCartOrder = await getCurrentCartOrder(ctx); + + // Check the lines and their options + const lineIndex = _.findIndex(currentCartOrder.lines, function(o) { + if (o.productId == productId) { + if (o.options && options) { + const from = JSON.parse(JSON.stringify(o.options)); + const to = JSON.parse(JSON.stringify(options)); + + return _.isEqual(from, to); + } else return true; + } + return false; + }); + + if (lineIndex === -1) { + // New Item + const _id = uuid(); + + // Verify if has parent ID + if (parentId) { + const parentIndex = _.findIndex(currentCartOrder.lines, { _id: parentId }); + if (parentIndex === -1) throw new ApolloError('parentProduct.notFound', '404'); + } + + // Verify if item exists + const productLine = plainToClass(LineItemDBSchema, { + _id, + ...input, + }); + + const product = await productLine.getProduct(ctx); + if (!product) throw new ApolloError('product.notFound', '404'); + + // Verify it's option + if (product.getLineProductFunctions().validateCartOption && input.options) + product.getLineProductFunctions().validateCartOption?.(input.options); + + const lineToAdd = productLine; //plainToClass(LineItemDBSchema, input); + currentCartOrder.lines.push(lineToAdd); + } else { + // Update the quantity + currentCartOrder.lines[lineIndex].quantity = quantity; + } + + const paymentIntentUpdated: PaymentIntentInfo = await updatePaymentIntent(currentCartOrder.paymentIntent, { + amount: await currentCartOrder.finalPrice(ctx), + currency: currentCartOrder.currency, + account, + }); + + return await orderModel.updateOneCustom({ + query: { _id: currentCartOrder._id }, + updateRequest: { + $set: { + paymentIntent: paymentIntentUpdated, + lines: currentCartOrder.lines, + }, + }, + ctx, + }); + } catch (error) { + throw error; + } +}; diff --git a/services/module-payments/functions/orders/services/LineService.ts b/services/module-payments/functions/orders/services/LineService.ts new file mode 100644 index 0000000..e69de29 diff --git a/services/module-payments/functions/orders/services/OrderService.ts b/services/module-payments/functions/orders/services/OrderService.ts new file mode 100644 index 0000000..52ffc16 --- /dev/null +++ b/services/module-payments/functions/orders/services/OrderService.ts @@ -0,0 +1,290 @@ +import { ApolloContext, ApolloContextLoadersOnly } from '@seed/interfaces/context'; +import { v4 as uuid } from 'uuid'; +import { ApolloError } from 'apollo-server-lambda'; +import _ from 'lodash'; +import { ProviderSchema } from '../schemas/provider.schema'; +import { OrderEngineDBSchema, OrderEngineSchema } from '../schemas/order.schema'; + +import { LocaleInfo, OrderBusinessStatusEnum, OrderStatusEnum } from '@services/module-payments/components/components.orders'; +import { CurrencyEnum, PaymentIntentInfo, PaymentStatusEnum, PaymentProvider } from '@services/module-payments/components/components.payments'; + +import { confirmPaymentIntent, createPaymentIntent } from '@services/module-payments/helpers/payments.helpers'; +import { CheckoutEngineInput } from '../schemas/order.schema.input'; + +import { LineProductInterface } from '../carts/schemas/line.schema'; +import { plainToClass } from 'class-transformer'; +import { getCurrentCartOrder } from './CartService'; +import { AvailableTranslation } from '@src/__components/components'; +import OrderModel from '@src/orders/order.model'; +import { LineItemSchema } from '@src/orders/components/line.schema'; +import { createApolloContext } from '@seed/graphql/Middleware'; +import { newError } from '@seed/helpers/Error'; +import OrderEngineModel from '../order.model'; +import { EnginePathComponent } from '@seed/interfaces/components'; + +interface ValidateCheckoutInput { + (input: OrderEngineSchema): Promise; +} + +export interface CheckoutInputInterface { + payment: PaymentProvider; + contactInfo: CheckoutEngineInput; + input: any; + ctx: ApolloContext; + + validateInputCheckout?: ValidateCheckoutInput; +} + +export interface finaliseCheckoutWithOrderInputInterface { + type: 'default' | 'event'; + payment: PaymentProvider; + contactInfo: CheckoutEngineInput; + input: any; + ctx: ApolloContext; +} + +export interface beginCheckoutWithOrderInputInterface { + contactInfo?: CheckoutEngineInput; + lines: LineProductInterface[]; + input: any; + ctx: ApolloContext; + paths?: EnginePathComponent[]; +} + +export interface CheckoutInterface { + payment: PaymentProvider; + cartOrderData: OrderEngineSchema; + contactInfo: CheckoutEngineInput; + input: any; +} + +// eslint-disable-next-line @typescript-eslint/explicit-function-return-type +export const setProviderData = async (orderData: OrderEngineSchema, ctx: ApolloContext, providersData: ProviderSchema[]) => { + for (let index = 0; index < orderData.lines.length; index++) { + const element = orderData.lines[index]; + const product = await element.getProduct(ctx); + + if (product?.organisationId) { + const index = _.findIndex(providersData, { organisationId: product.organisationId }); + if (index === -1) + providersData.push({ + _id: uuid(), + orderStatus: OrderBusinessStatusEnum.processing, + organisationId: product.organisationId, + lines: [element], + }); + else { + providersData[index].lines.push(element); + } + } + } +}; + +export const createOrder = async (data: { + lines?: LineProductInterface[]; + status?: OrderStatusEnum; + contactInfo?: CheckoutEngineInput; + currency?: CurrencyEnum; + localeInfo?: LocaleInfo; + input?: any; + paths?: EnginePathComponent[]; + ctx: ApolloContext; +}) => { + let { contactInfo, lines, currency, localeInfo, status, input, paths, ctx } = data; + const accountId = ctx.ctx.user._id; + + const currentLines = lines || []; + if (!currency) currency = (process.env.PAYMENT_DEFAULT_CURRENCY as CurrencyEnum) || CurrencyEnum.usd; + if (!status) status = OrderStatusEnum.draft; + if (!localeInfo) + localeInfo = { + countryCode: process.env.PAYMENT_DEFAULT_COUNTRY || 'BE', + locale: (process.env.PAYMENT_DEFAULT_LOCALE as any) || AvailableTranslation.en, + }; + + currentLines.forEach((line,key) => { + currentLines[key].localeInfo = localeInfo; + }); + + + + // (1) Create the order Info + const orderModel = new OrderModel({ + orderStatus: status, + accountId, + paymentIntent: { + provider: PaymentProvider.stripe, + }, + currency, + lines: plainToClass(LineItemSchema, currentLines), + localeInfo: localeInfo, + inputData: input, + + vatExempt: false, + + + ...contactInfo, + }); + + // (2) Get the amount & intent + const amount = await orderModel.dbData.finalPrice(ctx); + const paymentIntent: PaymentIntentInfo | undefined = await createPaymentIntent({ amount, currency, account: ctx.ctx.user }); + + if (!paymentIntent) throw newError(6000); + else { + return await orderModel.saveOne({ + additionnalData: { + paths, + paymentIntent, + }, + ctx, + }); + } +}; + +export const createDraftOrder = async (data: { currency?: CurrencyEnum; localeInfo?: LocaleInfo; input?: any; ctx: ApolloContext }) => { + return createOrder({ + status: OrderStatusEnum.draft, + ...data, + }); +}; + +export const checkout = async (checkoutData: CheckoutInterface): Promise => { + const { payment, cartOrderData, contactInfo, input } = checkoutData; + try { + if (!cartOrderData.lines || cartOrderData.lines.length == 0) throw newError(6003); + + let provider, + transactionId, + paymentStatus: PaymentStatusEnum, + orderStatus: OrderStatusEnum = OrderStatusEnum.draft; + + if (payment == PaymentProvider.stripe) { + if (!cartOrderData.paymentIntent.stripePaymentIntentData) throw newError(6004); + paymentStatus = PaymentStatusEnum.pending; + orderStatus = OrderStatusEnum.processing; + provider = PaymentProvider.stripe; + transactionId = cartOrderData.paymentIntent.stripePaymentIntentData.id; + } else { + if (payment == PaymentProvider.free) { + provider = PaymentProvider.bankWire; + // TODO + transactionId = ''; + paymentStatus = PaymentStatusEnum.free; + orderStatus = OrderStatusEnum.processing; + } else { + provider = PaymentProvider.bankWire; + // TODO + transactionId = ''; + paymentStatus = PaymentStatusEnum.pending; + orderStatus = OrderStatusEnum.processing; + } + } + + // Update the order data + const orderModel = new OrderModel(); + + const dataToSave: Partial = { + orderStatus, + billingInfo: contactInfo.billingInfo, + contactInfo: contactInfo.contactInfo, + paymentStatus: paymentStatus, + paymentInfo: { + provider, + transactionId, + }, + inputData: input, + }; + + const result = await orderModel.updateOne({ + query: { _id: cartOrderData._id }, + newData: dataToSave, + }); + + if (payment != PaymentProvider.stripe) await orderModel.afterPaymentSuccess(result); + + return result; + } catch (error) { + throw error; + } +}; + +export const createCheckout = async (data: CheckoutInputInterface): Promise => { + try { + const { validateInputCheckout, payment, ctx, contactInfo, input } = data; + // Get the cart + const cartOrderData = await getCurrentCartOrder(ctx); + if (!cartOrderData) throw new ApolloError('cart.notFound', '404'); + + // Validate input function + if (validateInputCheckout) validateInputCheckout(cartOrderData); + + return await checkout({ payment, cartOrderData, contactInfo, input }); + } catch (error) { + throw error; + } +}; + +export const beginCheckoutWithOrder = async (data: beginCheckoutWithOrderInputInterface): Promise => { + try { + const { contactInfo, lines, ctx, input, paths } = data; + + // Delete all previous cart + await (await new OrderEngineModel().db()).deleteMany({ accountId: ctx.ctx.user._id }); + + // Get the cart + const cartOrderData = await createOrder({ + contactInfo, + lines, + input: input, + paths, + ctx, + }); + + if (!cartOrderData) throw new ApolloError('cart.notFound', '404'); + + return cartOrderData; + } catch (error) { + throw error; + } +}; + +export const finaliseCheckoutWithOrder = async (data: finaliseCheckoutWithOrderInputInterface): Promise => { + try { + const { type, payment, contactInfo, ctx, input } = data; + + // Get the cart + const cartOrderData = await getCurrentCartOrder(ctx); + if (!cartOrderData) throw new ApolloError('cart.notFound', '404'); + + return await checkout({ payment, cartOrderData, contactInfo, input }); + } catch (error) { + throw error; + } +}; + +export async function localCheckout(orderData: OrderEngineSchema): Promise { + if (process.env.NODE_ENV === 'local') { + const orderModel = new OrderModel(); + try { + // Update the payment Intent and it's status + const dataToSave: Partial = { + paymentStatus: PaymentStatusEnum.paid, + orderStatus: OrderStatusEnum.processing, + }; + // check if organisations ids on product + const providersData: ProviderSchema[] = []; + await setProviderData(orderData, await createApolloContext(), providersData); + if (providersData.length > 0) dataToSave.providerOrderItems = providersData; + const order = await orderModel.updateOne({ + query: { _id: orderData._id }, + newData: dataToSave, + }); + await confirmPaymentIntent(orderData.paymentIntent); + await orderModel.afterPaymentSuccess(order); + } catch (error) { + console.error(error); + } + } + else throw newError(1002) +} diff --git a/services/module-payments/functions/paymentMethods/pm.resolver.admin.ts b/services/module-payments/functions/paymentMethods/pm.resolver.admin.ts new file mode 100644 index 0000000..a704264 --- /dev/null +++ b/services/module-payments/functions/paymentMethods/pm.resolver.admin.ts @@ -0,0 +1,14 @@ +import AccountModel from '@src/accounts/account.model'; +import { Resolver } from 'type-graphql'; + +@Resolver(AccountModel) +export default class AccountPaymentMethodAdminResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ +} diff --git a/services/module-payments/functions/paymentMethods/pm.resolver.fields.ts b/services/module-payments/functions/paymentMethods/pm.resolver.fields.ts new file mode 100644 index 0000000..c615ef4 --- /dev/null +++ b/services/module-payments/functions/paymentMethods/pm.resolver.fields.ts @@ -0,0 +1,14 @@ +import AccountModel from '@src/accounts/account.model'; +import { Resolver } from 'type-graphql'; + +@Resolver(AccountModel) +export default class AccountPaymentMethodFieldResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ +} diff --git a/services/module-payments/functions/paymentMethods/pm.resolver.ts b/services/module-payments/functions/paymentMethods/pm.resolver.ts new file mode 100644 index 0000000..77b150a --- /dev/null +++ b/services/module-payments/functions/paymentMethods/pm.resolver.ts @@ -0,0 +1,95 @@ +import { Resolver, Arg, Mutation, Ctx, Authorized } from 'type-graphql'; +import AccountModel from '@src/accounts/account.model'; +import { ApolloContext } from '@lib/seed/interfaces/context'; +import StripeService from '@services/module-payments/services/stripe/StripeService'; +import { ApolloError } from 'apollo-server-lambda'; +import { BillingAddressComponent } from '@services/module-payments/components/components'; +import { PaymentMethodDetail, PaymentMethodInput } from '@services/module-payments/components/components.payments'; +import { newError } from '@seed/helpers/Error'; + +@Resolver(AccountModel) +export default class AccountPaymentMethodResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + + @Mutation(() => PaymentMethodDetail) + @Authorized() + async accountsPaymentMethodsAddOne(@Arg('input') input: PaymentMethodInput, @Ctx() ctx: ApolloContext): Promise { + const model = ctx.ctx.user; + + // Get the card information from stripe + const pmResult = await StripeService.getInstance().createPaymentMethod(model, input.sPayMethodId); + + const cardDetail = { + ...input, + sPayMethodId: pmResult.id, + type: pmResult.type, + cardInfo: pmResult.card, + }; + + if (input.default && model.paymentInfo && model.paymentInfo.paymentMethods) { + // Remove all default from others + await model.updateOneCustom( + { _id: model._id }, + { + $set: { 'paymentInfo.paymentMethods.$[].default': false }, + }, + ctx, + ); + } else input.default = false; + + return await model.addOneSubRessource({ _id: model._id }, 'paymentInfo.paymentMethods', cardDetail, ctx); + } + + @Mutation(() => PaymentMethodDetail) + @Authorized() + async accountsPaymentMethodsMarkasDefault(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = ctx.ctx.user; + + // Remove all default from others + await model.updateOneCustom( + { _id: model._id, 'paymentInfo.paymentMethods._id': id }, + { + $set: { 'paymentInfo.paymentMethods.$[].default': false }, + }, + ctx, + ); + + await model.updateOneCustom( + { _id: model._id, 'paymentInfo.paymentMethods._id': id }, + { + $set: { 'paymentInfo.paymentMethods.$.default': true }, + }, + ctx, + ); + + return await model.getOneSubRessource('paymentInfo.paymentMethods', id); + } + + @Mutation(() => String) + @Authorized() + async accountsPaymentMethodsDeleteOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = ctx.ctx.user; + + // Get the PM id + const currentPM = model.getOneSubRessource('paymentInfo.paymentMethods', id) as PaymentMethodDetail; + if (!currentPM) throw newError(404); + + // Get the card information from stripe + return await model.deleteOneSubRessource({ _id: model._id }, 'paymentInfo.paymentMethods', id, ctx); + } +} diff --git a/services/module-payments/functions/payoutAccounts/pa.resolver.admin.ts b/services/module-payments/functions/payoutAccounts/pa.resolver.admin.ts new file mode 100644 index 0000000..7ce45ce --- /dev/null +++ b/services/module-payments/functions/payoutAccounts/pa.resolver.admin.ts @@ -0,0 +1,14 @@ +import AccountModel from '@src/accounts/account.model'; +import { Resolver } from 'type-graphql'; + +@Resolver(AccountModel) +export default class AccountPayoutAccountsAdminResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ +} diff --git a/services/module-payments/functions/payoutAccounts/pa.resolver.fields.ts b/services/module-payments/functions/payoutAccounts/pa.resolver.fields.ts new file mode 100644 index 0000000..1832899 --- /dev/null +++ b/services/module-payments/functions/payoutAccounts/pa.resolver.fields.ts @@ -0,0 +1,14 @@ +import AccountModel from '@src/accounts/account.model'; +import { Resolver } from 'type-graphql'; + +@Resolver(AccountModel) +export default class AccountPayoutAccountsFieldResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ +} diff --git a/services/module-payments/functions/payoutAccounts/pa.resolver.ts b/services/module-payments/functions/payoutAccounts/pa.resolver.ts new file mode 100644 index 0000000..186d3af --- /dev/null +++ b/services/module-payments/functions/payoutAccounts/pa.resolver.ts @@ -0,0 +1,61 @@ +import { Resolver, Arg, Mutation, Ctx, Authorized, Query, Args } from 'type-graphql'; +import AccountModel from '@src/accounts/account.model'; +import { ApolloContext } from '@lib/seed/interfaces/context'; +import StripeService from '@services/module-payments/services/stripe/StripeService'; + +@Resolver(AccountModel) +export default class AccountPayoutAccountsResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + @Query(() => String) + @Authorized() + async accountsDashboardLinkStripe(@Arg('returnPath', {nullable:true}) returnPath:string | undefined, @Ctx() ctx: ApolloContext): Promise { + const account = ctx.ctx.user; + + if (account.payoutInfo?.stripeInfo?.detailsSubmitted) { + // Get dashboard link + return await StripeService.getInstance().createAccountDashboardLink(account); + } + // Get Create Link + else return (await StripeService.getInstance().createAccountLink(account, returnPath, 'create')).url; + } + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + + @Mutation(() => String) + @Authorized() + async accountsLinkWithStripe(@Arg('returnPath', {nullable:true}) returnPath:string | undefined, @Ctx() ctx: ApolloContext): Promise { + const account = ctx.ctx.user; + + // Check if update of create of the link + let type: 'create' | 'update' = 'create'; + if (account.payoutInfo?.stripeInfo?.detailsSubmitted) type = 'update'; + + // Get the card information from stripe + const stripeAccountLink = await StripeService.getInstance().createAccountLink(account, returnPath, type); + + return stripeAccountLink.url; + } + + @Mutation(() => AccountModel) + @Authorized() + async accountsRefreshInfoStripe(@Ctx() ctx: ApolloContext): Promise { + const account = ctx.ctx.user; + + // Get the card information from stripe + return await StripeService.getInstance().refreshAccountInfo(account); + } +} diff --git a/services/module-payments/functions/promos/component.ts b/services/module-payments/functions/promos/component.ts new file mode 100644 index 0000000..58b80b9 --- /dev/null +++ b/services/module-payments/functions/promos/component.ts @@ -0,0 +1,9 @@ +import { registerEnumType } from 'type-graphql'; + +export enum PromoType { + fixed = 'fixed', + percentage = 'percentage', +} +registerEnumType(PromoType, { + name: 'PromoType', +}); diff --git a/services/module-payments/functions/promos/promo.model.ts b/services/module-payments/functions/promos/promo.model.ts new file mode 100644 index 0000000..a6ab3e0 --- /dev/null +++ b/services/module-payments/functions/promos/promo.model.ts @@ -0,0 +1,171 @@ +import { ObjectType, Field, ID, ArgsType, InputType, registerEnumType } from 'type-graphql'; +import { ApolloError } from 'apollo-server-lambda'; + +import { BaseGraphModel } from '@seed/graphql/BaseGraphModel'; +import { Permission } from '@seed/interfaces/permission'; + +import { GetArgs } from '@seed/graphql/Request'; + +import AccountModel from '@src/accounts/account.model'; +import { TranslatableComponent } from '@src/__components/components'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { ApolloContext } from '@seed/interfaces/context'; +import { PromoType } from './component'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.public, AccountTypeEnum.admin], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class PromoModel extends BaseGraphModel { + public constructor() { + super({ + collectionName: 'shop.promos', + permissions: permissions, + }); + } + @Field(() => ID) + readonly _id: string; + + @Field() + code: string; + + @Field(() => TranslatableComponent, { nullable: true }) + description: TranslatableComponent; + + @Field(() => PromoType) + type: PromoType; + + @Field() + value: number; + + @Field({ nullable: true }) + validity?: Date; + + @Field() + cummulable: boolean; + + @Field({ nullable: true }) + usageLimit?: number; + + @Field() + usage: number; + + searchOptions(): string[] { + return ['code']; + } + + filterOptions(): string[] { + return ['code']; + } + + // validatePromo = (): void => { + // try { + // if (this.usageLimit && this.usageLimit <= this.usage) { + // throw new ApolloError('promo.notValid', '404'); + // } + // if (this.validity) { + // const today = new Date(); + // if (this.validity < today) throw new ApolloError('promo.expired', '404'); + // } + // } catch (error) { + // throw error; + // } + // }; +} + +@ArgsType() +export class PromoArgs { + @Field(() => GetArgs, { nullable: true }) + pagination?: GetArgs; + @Field({ nullable: true }) + search?: string; +} + +@InputType() +export class NewPromoInput { + @Field() + code: string; + + @Field(() => TranslatableComponent, { nullable: true }) + description: TranslatableComponent; + + @Field(() => PromoType) + type: PromoType; + + @Field() + value: number; + + @Field() + cummulable: boolean; + + @Field({ nullable: true }) + validity?: Date; + + @Field({ nullable: true }) + usageLimit?: number; +} + +@InputType() +export class EditPromoInput { + @Field({ nullable: true }) + code: string; + + @Field(() => TranslatableComponent, { nullable: true }) + description: TranslatableComponent; + + @Field(() => PromoType, { nullable: true }) + type: PromoType; + + @Field({ nullable: true }) + value: number; + + @Field({ nullable: true }) + cummulable: boolean; + + @Field({ nullable: true }) + validity?: Date; + + @Field({ nullable: true }) + usageLimit?: number; +} + +export const onCreate = async (input: NewPromoInput, ctx: ApolloContext): Promise => { + try { + const promo = await (await new PromoModel().db()).findOne({ code: input.code }); + if (promo) { + throw new ApolloError('promo.nameTaken', '404'); + } + } catch (error) { + throw error; + } +}; + +export const onUpdate = async (input: EditPromoInput, ctx: ApolloContext): Promise => { + try { + const promo = await (await new PromoModel().db()).findOne({ code: input.code }); + if (promo) { + throw new ApolloError('promo.nameTaken', '404'); + } + } catch (error) { + throw error; + } +}; + +export const validatePromo = async (code: string, ctx: ApolloContext): Promise => { + try { + const promo = await new PromoModel().getOne({ code }, ctx); + if (promo.usageLimit && promo.usageLimit <= promo.usage) { + throw new ApolloError('promo.notValid', '404'); + } + if (promo.validity) { + const today = new Date(); + if (promo.validity < today) throw new ApolloError('promo.expired', '404'); + } + } catch (error) { + throw error; + } +}; diff --git a/services/module-payments/functions/promos/promo.resolver.admin.ts b/services/module-payments/functions/promos/promo.resolver.admin.ts new file mode 100644 index 0000000..501c597 --- /dev/null +++ b/services/module-payments/functions/promos/promo.resolver.admin.ts @@ -0,0 +1,51 @@ +import { Resolver, Query, FieldResolver, Root, Arg, Mutation, Ctx, Args, Authorized, ResolverInterface } from 'type-graphql'; + +import PromoModel, { PromoArgs, NewPromoInput, EditPromoInput } from '@services/module-payments/functions/promos/promo.model'; +import { ApolloContext } from '@seed/interfaces/context'; +import { createBaseResolver } from '@seed/graphql/baseResolvers/BaseResolver'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { getOneGeneric, getManyGenericWithArgs, addOneGeneric, editOneGeneric, deleteOneGeneric } from '@seed/graphql/BaseService'; +import { onCreate, onUpdate } from './promo.model'; + +const PromoBaseResolver = createBaseResolver('promos', PromoModel, PromoArgs, NewPromoInput, EditPromoInput, [AccountTypeEnum.admin]); + +@Resolver(PromoModel) +export default class PromoAdminResolver extends PromoBaseResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + @Mutation(() => PromoModel) + @Authorized([AccountTypeEnum.admin]) + async addOne(@Arg('input') input: NewPromoInput, @Ctx() ctx: ApolloContext): Promise { + const model = new PromoModel(); + const toSave = { + ...input, + usage: 0, + }; + await onCreate(input, ctx); + + return addOneGeneric(model, toSave, ctx); + } + + @Mutation(() => PromoModel) + @Authorized([AccountTypeEnum.admin]) + async editOne(@Arg('id') id: string, @Arg('input') input: EditPromoInput, @Ctx() ctx: ApolloContext): Promise { + const model = new PromoModel(); + await onUpdate(input, ctx); + + return editOneGeneric(model, id, input, ctx); + } +} diff --git a/services/module-payments/functions/promos/promo.resolver.ts b/services/module-payments/functions/promos/promo.resolver.ts new file mode 100644 index 0000000..2d9147e --- /dev/null +++ b/services/module-payments/functions/promos/promo.resolver.ts @@ -0,0 +1,32 @@ +import { Resolver, Query, FieldResolver, Root, Arg, Mutation, Ctx, Args, Authorized, ResolverInterface } from 'type-graphql'; + +import { getOneGeneric } from '@seed/graphql/BaseService'; +import PromoModel from '@services/module-payments/functions/promos/promo.model'; +import { ApolloContext } from '@seed/interfaces/context'; + +@Resolver(PromoModel) +export default class PromoResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + @Query(() => PromoModel) + async promosGetOneByCode(@Arg('code') code: string, @Ctx() ctx: ApolloContext): Promise { + const model = new PromoModel(); + return await model.getOne({ code }, ctx); + } + + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ +} diff --git a/services/module-payments/functions/vat/vat.components.ts b/services/module-payments/functions/vat/vat.components.ts new file mode 100644 index 0000000..4298758 --- /dev/null +++ b/services/module-payments/functions/vat/vat.components.ts @@ -0,0 +1,761 @@ +import { ObjectType, Field, InputType } from 'type-graphql'; + +@ObjectType() +@InputType('TVAValueInput') +export class TVAValue { + @Field(() => Number, { defaultValue: 21 }) + AF: number; + + @Field(() => Number, { defaultValue: 21 }) + AX: number; + + @Field(() => Number, { defaultValue: 21 }) + AL: number; + + @Field(() => Number, { defaultValue: 21 }) + DZ: number; + + @Field(() => Number, { defaultValue: 21 }) + AS: number; + + @Field(() => Number, { defaultValue: 21 }) + AD: number; + + @Field(() => Number, { defaultValue: 21 }) + AO: number; + + @Field(() => Number, { defaultValue: 21 }) + AI: number; + + @Field(() => Number, { defaultValue: 21 }) + AQ: number; + + @Field(() => Number, { defaultValue: 21 }) + AG: number; + + @Field(() => Number, { defaultValue: 21 }) + AR: number; + + @Field(() => Number, { defaultValue: 21 }) + AM: number; + + @Field(() => Number, { defaultValue: 21 }) + AW: number; + + @Field(() => Number, { defaultValue: 21 }) + AU: number; + + @Field(() => Number, { defaultValue: 21 }) + AT: number; + + @Field(() => Number, { defaultValue: 21 }) + AZ: number; + + @Field(() => Number, { defaultValue: 21 }) + BS: number; + + @Field(() => Number, { defaultValue: 21 }) + BH: number; + + @Field(() => Number, { defaultValue: 21 }) + BD: number; + + @Field(() => Number, { defaultValue: 21 }) + BB: number; + + @Field(() => Number, { defaultValue: 21 }) + BY: number; + + @Field(() => Number, { defaultValue: 21 }) + BE: number; + + @Field(() => Number, { defaultValue: 21 }) + BZ: number; + + @Field(() => Number, { defaultValue: 21 }) + BJ: number; + + @Field(() => Number, { defaultValue: 21 }) + BM: number; + + @Field(() => Number, { defaultValue: 21 }) + BT: number; + + @Field(() => Number, { defaultValue: 21 }) + BO: number; + + @Field(() => Number, { defaultValue: 21 }) + BA: number; + + @Field(() => Number, { defaultValue: 21 }) + BW: number; + + @Field(() => Number, { defaultValue: 21 }) + BV: number; + + @Field(() => Number, { defaultValue: 21 }) + BR: number; + + @Field(() => Number, { defaultValue: 21 }) + IO: number; + + @Field(() => Number, { defaultValue: 21 }) + BN: number; + + @Field(() => Number, { defaultValue: 21 }) + BG: number; + + @Field(() => Number, { defaultValue: 21 }) + BF: number; + + @Field(() => Number, { defaultValue: 21 }) + BI: number; + + @Field(() => Number, { defaultValue: 21 }) + KH: number; + + @Field(() => Number, { defaultValue: 21 }) + CM: number; + + @Field(() => Number, { defaultValue: 21 }) + CA: number; + + @Field(() => Number, { defaultValue: 21 }) + CV: number; + + @Field(() => Number, { defaultValue: 21 }) + KY: number; + + @Field(() => Number, { defaultValue: 21 }) + CF: number; + + @Field(() => Number, { defaultValue: 21 }) + TD: number; + + @Field(() => Number, { defaultValue: 21 }) + CL: number; + + @Field(() => Number, { defaultValue: 21 }) + CN: number; + + @Field(() => Number, { defaultValue: 21 }) + CX: number; + + @Field(() => Number, { defaultValue: 21 }) + CC: number; + + @Field(() => Number, { defaultValue: 21 }) + CO: number; + + @Field(() => Number, { defaultValue: 21 }) + KM: number; + + @Field(() => Number, { defaultValue: 21 }) + CG: number; + + @Field(() => Number, { defaultValue: 21 }) + CD: number; + + @Field(() => Number, { defaultValue: 21 }) + CK: number; + + @Field(() => Number, { defaultValue: 21 }) + CR: number; + + @Field(() => Number, { defaultValue: 21 }) + CI: number; + + @Field(() => Number, { defaultValue: 21 }) + HR: number; + + @Field(() => Number, { defaultValue: 21 }) + CU: number; + + @Field(() => Number, { defaultValue: 21 }) + CY: number; + + @Field(() => Number, { defaultValue: 21 }) + CZ: number; + + @Field(() => Number, { defaultValue: 21 }) + DK: number; + + @Field(() => Number, { defaultValue: 21 }) + DJ: number; + + @Field(() => Number, { defaultValue: 21 }) + DM: number; + + @Field(() => Number, { defaultValue: 21 }) + DO: number; + + @Field(() => Number, { defaultValue: 21 }) + EC: number; + + @Field(() => Number, { defaultValue: 21 }) + EG: number; + + @Field(() => Number, { defaultValue: 21 }) + SV: number; + + @Field(() => Number, { defaultValue: 21 }) + GQ: number; + + @Field(() => Number, { defaultValue: 21 }) + ER: number; + + @Field(() => Number, { defaultValue: 21 }) + EE: number; + + @Field(() => Number, { defaultValue: 21 }) + ET: number; + + @Field(() => Number, { defaultValue: 21 }) + FK: number; + + @Field(() => Number, { defaultValue: 21 }) + FO: number; + + @Field(() => Number, { defaultValue: 21 }) + FJ: number; + + @Field(() => Number, { defaultValue: 21 }) + FI: number; + + @Field(() => Number, { defaultValue: 21 }) + FR: number; + + @Field(() => Number, { defaultValue: 21 }) + GF: number; + + @Field(() => Number, { defaultValue: 21 }) + PF: number; + + @Field(() => Number, { defaultValue: 21 }) + TF: number; + + @Field(() => Number, { defaultValue: 21 }) + GA: number; + + @Field(() => Number, { defaultValue: 21 }) + GM: number; + + @Field(() => Number, { defaultValue: 21 }) + GE: number; + + @Field(() => Number, { defaultValue: 21 }) + DE: number; + + @Field(() => Number, { defaultValue: 21 }) + GH: number; + + @Field(() => Number, { defaultValue: 21 }) + GI: number; + + @Field(() => Number, { defaultValue: 21 }) + GR: number; + + @Field(() => Number, { defaultValue: 21 }) + GL: number; + + @Field(() => Number, { defaultValue: 21 }) + GD: number; + + @Field(() => Number, { defaultValue: 21 }) + GP: number; + + @Field(() => Number, { defaultValue: 21 }) + GU: number; + + @Field(() => Number, { defaultValue: 21 }) + GT: number; + + @Field(() => Number, { defaultValue: 21 }) + GG: number; + + @Field(() => Number, { defaultValue: 21 }) + GN: number; + + @Field(() => Number, { defaultValue: 21 }) + GW: number; + + @Field(() => Number, { defaultValue: 21 }) + GY: number; + + @Field(() => Number, { defaultValue: 21 }) + HT: number; + + @Field(() => Number, { defaultValue: 21 }) + HM: number; + + @Field(() => Number, { defaultValue: 21 }) + VA: number; + + @Field(() => Number, { defaultValue: 21 }) + HN: number; + + @Field(() => Number, { defaultValue: 21 }) + HK: number; + + @Field(() => Number, { defaultValue: 21 }) + HU: number; + + @Field(() => Number, { defaultValue: 21 }) + IS: number; + + @Field(() => Number, { defaultValue: 21 }) + IN: number; + + @Field(() => Number, { defaultValue: 21 }) + ID: number; + + @Field(() => Number, { defaultValue: 21 }) + IR: number; + + @Field(() => Number, { defaultValue: 21 }) + IQ: number; + + @Field(() => Number, { defaultValue: 21 }) + IE: number; + + @Field(() => Number, { defaultValue: 21 }) + IM: number; + + @Field(() => Number, { defaultValue: 21 }) + IL: number; + + @Field(() => Number, { defaultValue: 21 }) + IT: number; + + @Field(() => Number, { defaultValue: 21 }) + JM: number; + + @Field(() => Number, { defaultValue: 21 }) + JP: number; + + @Field(() => Number, { defaultValue: 21 }) + JE: number; + + @Field(() => Number, { defaultValue: 21 }) + JO: number; + + @Field(() => Number, { defaultValue: 21 }) + KZ: number; + + @Field(() => Number, { defaultValue: 21 }) + KE: number; + + @Field(() => Number, { defaultValue: 21 }) + KI: number; + + @Field(() => Number, { defaultValue: 21 }) + KP: number; + + @Field(() => Number, { defaultValue: 21 }) + KR: number; + + @Field(() => Number, { defaultValue: 21 }) + KW: number; + + @Field(() => Number, { defaultValue: 21 }) + KG: number; + + @Field(() => Number, { defaultValue: 21 }) + LA: number; + + @Field(() => Number, { defaultValue: 21 }) + LV: number; + + @Field(() => Number, { defaultValue: 21 }) + LB: number; + + @Field(() => Number, { defaultValue: 21 }) + LS: number; + + @Field(() => Number, { defaultValue: 21 }) + LR: number; + + @Field(() => Number, { defaultValue: 21 }) + LY: number; + + @Field(() => Number, { defaultValue: 21 }) + LI: number; + + @Field(() => Number, { defaultValue: 21 }) + LT: number; + + @Field(() => Number, { defaultValue: 21 }) + LU: number; + + @Field(() => Number, { defaultValue: 21 }) + MO: number; + + @Field(() => Number, { defaultValue: 21 }) + MK: number; + + @Field(() => Number, { defaultValue: 21 }) + MG: number; + + @Field(() => Number, { defaultValue: 21 }) + MW: number; + + @Field(() => Number, { defaultValue: 21 }) + MY: number; + + @Field(() => Number, { defaultValue: 21 }) + MV: number; + + @Field(() => Number, { defaultValue: 21 }) + ML: number; + + @Field(() => Number, { defaultValue: 21 }) + MT: number; + + @Field(() => Number, { defaultValue: 21 }) + MH: number; + + @Field(() => Number, { defaultValue: 21 }) + MQ: number; + + @Field(() => Number, { defaultValue: 21 }) + MR: number; + + @Field(() => Number, { defaultValue: 21 }) + MU: number; + + @Field(() => Number, { defaultValue: 21 }) + YT: number; + + @Field(() => Number, { defaultValue: 21 }) + MX: number; + + @Field(() => Number, { defaultValue: 21 }) + FM: number; + + @Field(() => Number, { defaultValue: 21 }) + MD: number; + + @Field(() => Number, { defaultValue: 21 }) + MC: number; + + @Field(() => Number, { defaultValue: 21 }) + MN: number; + + @Field(() => Number, { defaultValue: 21 }) + MS: number; + + @Field(() => Number, { defaultValue: 21 }) + MA: number; + + @Field(() => Number, { defaultValue: 21 }) + MZ: number; + + @Field(() => Number, { defaultValue: 21 }) + MM: number; + + @Field(() => Number, { defaultValue: 21 }) + NA: number; + + @Field(() => Number, { defaultValue: 21 }) + NR: number; + + @Field(() => Number, { defaultValue: 21 }) + NP: number; + + @Field(() => Number, { defaultValue: 21 }) + NL: number; + + @Field(() => Number, { defaultValue: 21 }) + AN: number; + + @Field(() => Number, { defaultValue: 21 }) + NC: number; + + @Field(() => Number, { defaultValue: 21 }) + NZ: number; + + @Field(() => Number, { defaultValue: 21 }) + NI: number; + + @Field(() => Number, { defaultValue: 21 }) + NE: number; + + @Field(() => Number, { defaultValue: 21 }) + NG: number; + + @Field(() => Number, { defaultValue: 21 }) + NU: number; + + @Field(() => Number, { defaultValue: 21 }) + NF: number; + + @Field(() => Number, { defaultValue: 21 }) + MP: number; + + @Field(() => Number, { defaultValue: 21 }) + NO: number; + + @Field(() => Number, { defaultValue: 21 }) + OM: number; + + @Field(() => Number, { defaultValue: 21 }) + PK: number; + + @Field(() => Number, { defaultValue: 21 }) + PW: number; + + @Field(() => Number, { defaultValue: 21 }) + PS: number; + + @Field(() => Number, { defaultValue: 21 }) + PA: number; + + @Field(() => Number, { defaultValue: 21 }) + PG: number; + + @Field(() => Number, { defaultValue: 21 }) + PY: number; + + @Field(() => Number, { defaultValue: 21 }) + PE: number; + + @Field(() => Number, { defaultValue: 21 }) + PH: number; + + @Field(() => Number, { defaultValue: 21 }) + PN: number; + + @Field(() => Number, { defaultValue: 21 }) + PL: number; + + @Field(() => Number, { defaultValue: 21 }) + PT: number; + + @Field(() => Number, { defaultValue: 21 }) + PR: number; + + @Field(() => Number, { defaultValue: 21 }) + QA: number; + + @Field(() => Number, { defaultValue: 21 }) + RE: number; + + @Field(() => Number, { defaultValue: 21 }) + RO: number; + + @Field(() => Number, { defaultValue: 21 }) + RU: number; + + @Field(() => Number, { defaultValue: 21 }) + RW: number; + + @Field(() => Number, { defaultValue: 21 }) + SH: number; + + @Field(() => Number, { defaultValue: 21 }) + KN: number; + + @Field(() => Number, { defaultValue: 21 }) + LC: number; + + @Field(() => Number, { defaultValue: 21 }) + PM: number; + + @Field(() => Number, { defaultValue: 21 }) + VC: number; + + @Field(() => Number, { defaultValue: 21 }) + WS: number; + + @Field(() => Number, { defaultValue: 21 }) + SM: number; + + @Field(() => Number, { defaultValue: 21 }) + ST: number; + + @Field(() => Number, { defaultValue: 21 }) + SA: number; + + @Field(() => Number, { defaultValue: 21 }) + SN: number; + + @Field(() => Number, { defaultValue: 21 }) + CS: number; + + @Field(() => Number, { defaultValue: 21 }) + SC: number; + + @Field(() => Number, { defaultValue: 21 }) + SL: number; + + @Field(() => Number, { defaultValue: 21 }) + SG: number; + + @Field(() => Number, { defaultValue: 21 }) + SK: number; + + @Field(() => Number, { defaultValue: 21 }) + SI: number; + + @Field(() => Number, { defaultValue: 21 }) + SB: number; + + @Field(() => Number, { defaultValue: 21 }) + SO: number; + + @Field(() => Number, { defaultValue: 21 }) + ZA: number; + + @Field(() => Number, { defaultValue: 21 }) + GS: number; + + @Field(() => Number, { defaultValue: 21 }) + ES: number; + + @Field(() => Number, { defaultValue: 21 }) + LK: number; + + @Field(() => Number, { defaultValue: 21 }) + SD: number; + + @Field(() => Number, { defaultValue: 21 }) + SR: number; + + @Field(() => Number, { defaultValue: 21 }) + SJ: number; + + @Field(() => Number, { defaultValue: 21 }) + SZ: number; + + @Field(() => Number, { defaultValue: 21 }) + SE: number; + + @Field(() => Number, { defaultValue: 21 }) + CH: number; + + @Field(() => Number, { defaultValue: 21 }) + SY: number; + + @Field(() => Number, { defaultValue: 21 }) + TW: number; + + @Field(() => Number, { defaultValue: 21 }) + TJ: number; + + @Field(() => Number, { defaultValue: 21 }) + TZ: number; + + @Field(() => Number, { defaultValue: 21 }) + TH: number; + + @Field(() => Number, { defaultValue: 21 }) + TL: number; + + @Field(() => Number, { defaultValue: 21 }) + TG: number; + + @Field(() => Number, { defaultValue: 21 }) + TK: number; + + @Field(() => Number, { defaultValue: 21 }) + TO: number; + + @Field(() => Number, { defaultValue: 21 }) + TT: number; + + @Field(() => Number, { defaultValue: 21 }) + TN: number; + + @Field(() => Number, { defaultValue: 21 }) + TR: number; + + @Field(() => Number, { defaultValue: 21 }) + TM: number; + + @Field(() => Number, { defaultValue: 21 }) + TC: number; + + @Field(() => Number, { defaultValue: 21 }) + TV: number; + + @Field(() => Number, { defaultValue: 21 }) + UG: number; + + @Field(() => Number, { defaultValue: 21 }) + UA: number; + + @Field(() => Number, { defaultValue: 21 }) + AE: number; + + @Field(() => Number, { defaultValue: 21 }) + GB: number; + + @Field(() => Number, { defaultValue: 21 }) + US: number; + + @Field(() => Number, { defaultValue: 21 }) + UM: number; + + @Field(() => Number, { defaultValue: 21 }) + UY: number; + + @Field(() => Number, { defaultValue: 21 }) + UZ: number; + + @Field(() => Number, { defaultValue: 21 }) + VU: number; + + @Field(() => Number, { defaultValue: 21 }) + VE: number; + + @Field(() => Number, { defaultValue: 21 }) + VN: number; + + @Field(() => Number, { defaultValue: 21 }) + VG: number; + + @Field(() => Number, { defaultValue: 21 }) + VI: number; + + @Field(() => Number, { defaultValue: 21 }) + WF: number; + + @Field(() => Number, { defaultValue: 21 }) + EH: number; + + @Field(() => Number, { defaultValue: 21 }) + YE: number; + + @Field(() => Number, { defaultValue: 21 }) + ZM: number; + + @Field(() => Number, { defaultValue: 21 }) + ZW: number; +} + +@ObjectType() +export class VatResponse { + @Field(() => Boolean) + valid: boolean; + + @Field({ nullable: true }) + database?: string; + + @Field(() => Boolean) + format_valid?: boolean; + + @Field({ nullable: true }) + query?: string; + + @Field({ nullable: true }) + country_code?: string; + + @Field({ nullable: true }) + vat_number?: string; + + @Field({ nullable: true }) + company_name?: string; + + @Field({ nullable: true }) + company_address?: string; +} diff --git a/services/module-payments/functions/vat/vat.model.ts b/services/module-payments/functions/vat/vat.model.ts new file mode 100644 index 0000000..40acbe8 --- /dev/null +++ b/services/module-payments/functions/vat/vat.model.ts @@ -0,0 +1,104 @@ +import { ObjectType, Field, ID, ArgsType, InputType, registerEnumType, Ctx } from 'type-graphql'; + +import { BaseGraphModel } from '@seed/graphql/BaseGraphModel'; +import { Permission } from '@seed/interfaces/permission'; +import { TranslatableComponent } from '@src/__components/components'; +import { GetArgs } from '@seed/graphql/Request'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { ApolloContext } from '@seed/interfaces/context'; +import { TVAValue } from './vat.components'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.admin, AccountTypeEnum.public], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class VatClassModel extends BaseGraphModel { + public constructor() { + super({ + // ...init, + collectionName: 'shop.vat', + permissions: permissions, + }); + } + // eslint-disable-next-line @typescript-eslint/no-unused-vars + @Field(() => ID) + readonly _id: string; + + @Field() + title: string; + + @Field(() => TVAValue) + values: TVAValue; + + searchOptions(): string[] { + return []; + } + + filterOptions(): string[] { + return []; + } +} + +@ArgsType() +export class VatClassArgs { + @Field(() => GetArgs, { nullable: true }) + pagination?: GetArgs; +} + +@InputType() +export class EditVatClassInput { + @Field() + title: string; + + @Field(() => TVAValue, { nullable: true }) + values?: TVAValue; +} + +@InputType() +export class NewVatClassInput { + @Field() + title: string; + + @Field(() => TVAValue) + values: TVAValue; +} + +@InputType() +export class ValidateVatInput { + @Field() + countryCode: string; + + @Field() + vatNumber: string; +} + +@ObjectType() +export class VatResponse { + @Field(() => Boolean) + valid: boolean; + + @Field({ nullable: true }) + database?: string; + + @Field(() => Boolean) + format_valid?: boolean; + + @Field({ nullable: true }) + query?: string; + + @Field({ nullable: true }) + country_code?: string; + + @Field({ nullable: true }) + vat_number?: string; + + @Field({ nullable: true }) + company_name?: string; + + @Field({ nullable: true }) + company_address?: string; +} diff --git a/services/module-payments/functions/vat/vat.resolver.admin.ts b/services/module-payments/functions/vat/vat.resolver.admin.ts new file mode 100644 index 0000000..8ac65e8 --- /dev/null +++ b/services/module-payments/functions/vat/vat.resolver.admin.ts @@ -0,0 +1,71 @@ +import { Resolver, Query, FieldResolver, Root, Arg, Mutation, Ctx, Args, Authorized, ResolverInterface } from 'type-graphql'; +import { v4 as uuid } from 'uuid'; +import _ from 'lodash'; +import { FilterQuery } from 'mongodb'; +import { ApolloError } from 'apollo-server-lambda'; + +import VatClassModel, { VatClassArgs, NewVatClassInput, EditVatClassInput } from './vat.model'; +import { ApolloContext } from '@seed/interfaces/context'; +import { createBaseResolver } from '@seed/graphql/baseResolvers/BaseResolver'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { getOneGeneric, addOneGeneric, editOneGeneric, deleteOneGeneric, permissionsAddOne, permissionsRemoveOne } from '@seed/graphql/BaseService'; + +// const VatBaseResolver = createBaseResolver('Vat', VatModel, VatArgs, NewVatInput, EditVatInput, [AccountTypeEnum.admin]); + +@Resolver(VatClassModel) +export default class VatAdminResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + @Query(() => VatClassModel) + @Authorized([AccountTypeEnum.admin]) + async vatClassGetOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = new VatClassModel(); + return getOneGeneric(model, id, ctx); + } + + @Query(() => [VatClassModel]) + @Authorized([AccountTypeEnum.admin]) + async vatClassGetMany(@Args() args: VatClassArgs, @Ctx() ctx: ApolloContext): Promise { + const model = new VatClassModel(); + const q: FilterQuery = {}; + + return model.getMany(q, args.pagination, ctx); + } + + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + + @Mutation(() => VatClassModel) + @Authorized([AccountTypeEnum.admin]) + async vatClassAddOne(@Arg('input') input: NewVatClassInput, @Ctx() ctx: ApolloContext): Promise { + const model = new VatClassModel(); + return addOneGeneric(model, input, ctx); + } + + @Mutation(() => VatClassModel) + @Authorized([AccountTypeEnum.admin]) + async vatClassEditOne(@Arg('id') id: string, @Arg('input') input: EditVatClassInput, @Ctx() ctx: ApolloContext): Promise { + const model = new VatClassModel(); + return editOneGeneric(model, id, input, ctx); + } + + @Mutation(() => VatClassModel) + @Authorized([AccountTypeEnum.admin]) + async vatClassDeleteOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = new VatClassModel(); + return deleteOneGeneric(model, id, ctx); + } +} diff --git a/services/module-payments/functions/vat/vat.resolver.ts b/services/module-payments/functions/vat/vat.resolver.ts new file mode 100644 index 0000000..8148608 --- /dev/null +++ b/services/module-payments/functions/vat/vat.resolver.ts @@ -0,0 +1,89 @@ +import { Resolver, Query, FieldResolver, Root, Arg, Mutation, Ctx, Args, Authorized, ResolverInterface } from 'type-graphql'; + +import VatClassModel from './vat.model'; +import OrderEngineModel from '@services/module-payments/functions/orders/order.model'; +import { ApolloContext } from '@seed/interfaces/context'; +import { ValidateVatInput, VatResponse } from './vat.model'; +import axios from 'axios'; +import { updatePaymentIntent } from '@services/module-payments/helpers/payments.helpers'; +import { PaymentIntentInfo } from '@services/module-payments/components/components.payments'; +import { getCurrentCartOrder } from '../orders/services/CartService'; + +@Resolver(VatClassModel) +export default class VatClassResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + @Mutation(() => Boolean) + @Authorized() + async vatValidate(@Arg('input') input: ValidateVatInput, @Ctx() ctx: ApolloContext): Promise { + //VATLAYER + const baseUrl = 'http://apilayer.net/api/validate?'; + const account = ctx.ctx.user; + const orderModel = new OrderEngineModel(); + + try { + const currentCartOrder = await getCurrentCartOrder(ctx); + + let valid = false; + let vatExempt = false; + + if (input.countryCode === process.env.SHOP_COUNTRY_CODE) valid = true; + else { + const vatResponse = await axios.get( + `${baseUrl}access_key=${process.env.VATLAYER_API_KEY}&vat_number=${input.countryCode}${input.vatNumber}`, + ); + if (vatResponse.data.valid) { + vatExempt = true; + valid = true; + } + } + + //Pas top de faire 2 requetes + + await orderModel.updateOneCustom({ + query: { _id: currentCartOrder._id }, + updateRequest: { + $set: { + vatExempt: vatExempt, + }, + }, + ctx, + }); + + //ne marche pas on dirait + const paymentIntentUpdated: PaymentIntentInfo = await updatePaymentIntent(currentCartOrder.paymentIntent, { + amount: await currentCartOrder.finalPrice(ctx), + currency: currentCartOrder.currency, + account, + }); + + await orderModel.updateOneCustom({ + query: { _id: currentCartOrder._id }, + updateRequest: { + $set: { + paymentIntent: paymentIntentUpdated, + }, + }, + ctx, + }); + + return valid; + } catch (error) { + throw error; + } + } +} diff --git a/services/module-payments/helpers/payments.helpers.ts b/services/module-payments/helpers/payments.helpers.ts new file mode 100644 index 0000000..fcf53b9 --- /dev/null +++ b/services/module-payments/helpers/payments.helpers.ts @@ -0,0 +1,93 @@ +import AccountModel from '@src/accounts/account.model'; +import { CurrencyEnum, PaymentIntentInfo, PaymentProvider } from '../components/components.payments'; +import StripeService from '../services/stripe/StripeService'; + +export const createPaymentIntent = async (input: { + amount?: number; + currency?: CurrencyEnum; + account?: AccountModel; +}): Promise => { + const { amount, currency, account } = input; + const currentPaymentProvider = process.env.PAYMENT_PROVIDER || 'stripe'; + + let paymentIntent: PaymentIntentInfo | undefined; + switch (currentPaymentProvider) { + case 'stripe': + const inputData: any = { + currency: currency || (process.env.PAYMENT_DEFAULT_CURRENCY as CurrencyEnum) || CurrencyEnum.usd, + }; + + if (amount) inputData.amount = amount; + if (account) inputData.account = account; + + // Create the PaymentIntent + const stripePaymentInfo = await StripeService.getInstance().createPaymentIntent(inputData); + + paymentIntent = { + provider: PaymentProvider.stripe, + stripePaymentIntentData: stripePaymentInfo as any, + }; + break; + + default: + break; + } + return paymentIntent; +}; + +export const updatePaymentIntent = async ( + paymentIntent: PaymentIntentInfo, + updateInfo: { + amount?: number; + currency?: CurrencyEnum; + account?: AccountModel; + }, +): Promise => { + const currentPaymentProvider = process.env.PAYMENT_PROVIDER || 'stripe'; + + switch (currentPaymentProvider) { + case 'stripe': + // Update the payment Intent + + if (paymentIntent.stripePaymentIntentData) { + const stripePaymentInfo = await StripeService.getInstance().updatePaymentIntent(paymentIntent.stripePaymentIntentData.id, updateInfo); + + paymentIntent = { + provider: PaymentProvider.stripe, + stripePaymentIntentData: stripePaymentInfo as any, + }; + } + + break; + + default: + break; + } + return paymentIntent; +}; + +export const confirmPaymentIntent = async (paymentIntent: PaymentIntentInfo): Promise => { + const currentPaymentProvider = process.env.PAYMENT_PROVIDER || 'stripe'; + + switch (currentPaymentProvider) { + case 'stripe': + // Update the payment Intent + + if (paymentIntent.stripePaymentIntentData) { + const stripePaymentInfo = await StripeService.getInstance().confirmPaymentIntent(paymentIntent.stripePaymentIntentData.id, { + paymentMethod: 'pm_card_visa', + }); + + paymentIntent = { + provider: PaymentProvider.stripe, + stripePaymentIntentData: stripePaymentInfo as any, + }; + } + + break; + + default: + break; + } + // return paymentIntent; +}; diff --git a/services/module-payments/index.ts b/services/module-payments/index.ts new file mode 100644 index 0000000..7f158bb --- /dev/null +++ b/services/module-payments/index.ts @@ -0,0 +1,9 @@ +import 'reflect-metadata'; +import AccountBillingInfoResolver from '@services/module-payments/functions/billingInfos/billingInfo.resolver'; +import AccountPaymentMethodResolver from '@services/module-payments/functions/paymentMethods/pm.resolver'; +import AccountPayoutAccountsResolver from '@services/module-payments/functions/payoutAccounts/pa.resolver'; + +export const PaymentAppEngineResolvers = [AccountBillingInfoResolver, AccountPaymentMethodResolver]; +export const PaymentAdminEngineResolvers = []; + +export const PayoutAppEngineResolvers = [AccountPayoutAccountsResolver]; diff --git a/services/module-payments/serverless-payments.yaml b/services/module-payments/serverless-payments.yaml new file mode 100644 index 0000000..20c2d67 --- /dev/null +++ b/services/module-payments/serverless-payments.yaml @@ -0,0 +1,46 @@ +app: ${file(./package.json):name} +service: payments +package: + individually: true +provider: + name: aws + stage: ${opt:stage,'dev'} + runtime: nodejs12.x + environment: + AWS_Hooks: hooks-${self:provider.stage}-hookHandler + iamRoleStatements: + - Effect: Allow + Action: + - lambda:InvokeFunction + Resource: '*' +plugins: + - serverless-webpack + - serverless-domain-manager + - serverless-offline +custom: + stage: ${opt:stage, self:provider.stage} + domains: ${file(./config/domains.yaml):domains} + customDomain: ${file(./config/domains.yaml):customDomain} + serverless-offline: + httpPort: 4002 + lambdaPort: 4010 + webpack: + webpackConfig: './lib/seed/webpack.config.js' # Name of webpack configuration file + includeModules: + forceExclude: + - aws-sdk + - puppeteer +functions: + stripe-hooks: + handler: handler.handler + timeout: 300 + events: + - http: + path: '/webhook' + method: post + cors: + origin: '*' + headers: + - Content-Type + - Authorization + - stripe-signature diff --git a/services/module-payments/services/stripe/StripeService.ts b/services/module-payments/services/stripe/StripeService.ts new file mode 100644 index 0000000..99ad210 --- /dev/null +++ b/services/module-payments/services/stripe/StripeService.ts @@ -0,0 +1,382 @@ +/* eslint-disable @typescript-eslint/camelcase */ +import Stripe from 'stripe'; +import { ApolloError } from 'apollo-server-lambda'; +import AccountModel from '@src/accounts/account.model'; +import _ from 'lodash'; +import { updateStripePayoutInfo } from './paymentIntent.helper'; +import { CurrencyEnum } from '@services/module-payments/components/components.payments'; +import { newError } from '@seed/helpers/Error'; + +export default class StripeService { + private static instance: StripeService; + public stripe: Stripe; + + private constructor() { + try { + this.stripe = new Stripe(process.env.STRIPE_SKEY || '', { maxNetworkRetries: 3, apiVersion: '2020-08-27' }); + } catch (error) { + console.error('STRIPE initialization error raised', error.stack); + } + } + + public static getInstance(): StripeService { + if (!StripeService.instance) { + StripeService.instance = new StripeService(); + } + return StripeService.instance; + } + + /* + ██████╗██╗ ██╗███████╗████████╗ ██████╗ ███╗ ███╗███████╗██████╗ ███████╗ + ██╔════╝██║ ██║██╔════╝╚══██╔══╝██╔═══██╗████╗ ████║██╔════╝██╔══██╗██╔════╝ + ██║ ██║ ██║███████╗ ██║ ██║ ██║██╔████╔██║█████╗ ██████╔╝███████╗ + ██║ ██║ ██║╚════██║ ██║ ██║ ██║██║╚██╔╝██║██╔══╝ ██╔══██╗╚════██║ + ╚██████╗╚██████╔╝███████║ ██║ ╚██████╔╝██║ ╚═╝ ██║███████╗██║ ██║███████║ + ╚═════╝ ╚═════╝ ╚══════╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝╚═╝ ╚═╝╚══════╝ + */ + + public createCustomer = async (account: AccountModel): Promise => { + if (account.paymentInfo && account.paymentInfo.stripeInfo) return account.paymentInfo.stripeInfo.customerId; + + try { + const stripeCreateCustomer = await this.stripe.customers.create({ + email: account.email, + metadata: { + _id: account._id, + }, + name: account.firstName + ' ' + account.lastName, + }); + if (stripeCreateCustomer.id) { + await account.updateOneCustom({ _id: account._id }, { $set: { 'paymentInfo.stripeInfo.customerId': stripeCreateCustomer.id } }, null); + return stripeCreateCustomer.id; + } else throw stripeCreateCustomer; + } catch (error) { + throw newError(6000, { message: error.message }); + } + }; + + /* + █████╗ ██████╗ ██████╗ ██████╗ ██╗ ██╗███╗ ██╗████████╗███████╗ + ██╔══██╗██╔════╝██╔════╝██╔═══██╗██║ ██║████╗ ██║╚══██╔══╝██╔════╝ + ███████║██║ ██║ ██║ ██║██║ ██║██╔██╗ ██║ ██║ ███████╗ + ██╔══██║██║ ██║ ██║ ██║██║ ██║██║╚██╗██║ ██║ ╚════██║ + ██║ ██║╚██████╗╚██████╗╚██████╔╝╚██████╔╝██║ ╚████║ ██║ ███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═════╝ ╚═════╝ ╚═════╝ ╚═╝ ╚═══╝ ╚═╝ ╚══════╝ + + */ + + public createAccount = async (account: AccountModel): Promise => { + if (account.payoutInfo?.stripeInfo) return account; + + try { + const stripeCreateAccount = await this.stripe.accounts.create({ + type: 'express', + email: account.email, + }); + if (stripeCreateAccount.id) { + return await updateStripePayoutInfo(account, stripeCreateAccount); + } else throw stripeCreateAccount; + } catch (error) { + throw newError(6000, { message: error.message }); + } + }; + + public refreshAccountInfo = async (account: AccountModel): Promise => { + if (!account.payoutInfo?.stripeInfo?.accountId) throw newError(6001); + + try { + const stripeAccount = await this.stripe.accounts.retrieve(account.payoutInfo?.stripeInfo?.accountId); + if (stripeAccount.id) { + return await updateStripePayoutInfo(account, stripeAccount); + } else throw stripeAccount; + } catch (error) { + throw newError(6000, { message: error.message }); + } + }; + + public createAccountLink = async (account: AccountModel, returnPath: string | undefined, reason?: 'create' | 'update'): Promise => { + try { + const stripeAccount = await this.createAccount(account); + + if (stripeAccount.payoutInfo?.stripeInfo?.accountId) { + let type: Stripe.AccountLinkCreateParams.Type = 'account_onboarding'; + if (reason && reason == 'update') type = 'account_update'; + + let returnUrl = process.env.STRIPE_ACCOUNT_RETURN_URL || ''; + if (returnPath && returnPath != "") { + while(returnPath.charAt(0) === '/') + { + returnPath = returnPath.substring(1); + } + + returnUrl = process.env.STRIPE_ACCOUNT_REFRESH_URL + "/" + returnPath + } + + return await this.stripe.accountLinks.create({ + account: stripeAccount.payoutInfo?.stripeInfo?.accountId, + refresh_url: process.env.STRIPE_ACCOUNT_REFRESH_URL || '', + return_url: returnUrl, + type: type, + }); + } else throw newError(6001); + } catch (error) { + throw error; + } + }; + + public createAccountDashboardLink = async (account: AccountModel): Promise => { + try { + const stripeAccount = await this.createAccount(account); + + if (stripeAccount.payoutInfo?.stripeInfo?.accountId) { + return await (await this.stripe.accounts.createLoginLink(stripeAccount.payoutInfo?.stripeInfo?.accountId)).url; + } else throw newError(6001); + } catch (error) { + throw error; + } + }; + + /* + ███╗ ███╗███████╗████████╗██╗ ██╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██╔════╝╚══██╔══╝██║ ██║██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║█████╗ ██║ ███████║██║ ██║██║ ██║███████╗ + ██║╚██╔╝██║██╔══╝ ██║ ██╔══██║██║ ██║██║ ██║╚════██║ + ██║ ╚═╝ ██║███████╗ ██║ ██║ ██║╚██████╔╝██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═════╝ ╚══════╝ + */ + + public createPaymentMethod = async (account: AccountModel, paymentMethodId: string): Promise => { + let customerId; + // Check if customer exists + if (account.paymentInfo && account.paymentInfo.stripeInfo) { + customerId = account.paymentInfo.stripeInfo.customerId; + } else customerId = await this.createCustomer(account); + + try { + // Check if that card exists already + + const paymentMethod = await this.stripe.paymentMethods.attach(paymentMethodId, { + customer: customerId, + }); + + return paymentMethod; + } catch (error) { + throw newError(6000, { message: error.message }); + } + }; + + public deletePaymentMethod = async (paymentMethodId: string): Promise => { + try { + // Check if that card exists already + + const paymentMethod = await this.stripe.paymentMethods.detach(paymentMethodId); + + return paymentMethod; + } catch (error) { + throw newError(6000, { message: error.message }); + } + }; + + /* + ██╗███╗ ██╗████████╗███████╗███╗ ██╗████████╗███████╗ + ██║████╗ ██║╚══██╔══╝██╔════╝████╗ ██║╚══██╔══╝██╔════╝ + ██║██╔██╗ ██║ ██║ █████╗ ██╔██╗ ██║ ██║ ███████╗ + ██║██║╚██╗██║ ██║ ██╔══╝ ██║╚██╗██║ ██║ ╚════██║ + ██║██║ ╚████║ ██║ ███████╗██║ ╚████║ ██║ ███████║ + ╚═╝╚═╝ ╚═══╝ ╚═╝ ╚══════╝╚═╝ ╚═══╝ ╚═╝ ╚══════╝ + + */ + + public createPaymentIntent = async (input: { + amount?: number; + currency?: CurrencyEnum; + paymentMethod?: string; + account?: AccountModel; + }): Promise => { + const { amount, currency, account, paymentMethod } = input; + + let customerId; + if (account) { + // Check if customer exists + if (account.paymentInfo && account.paymentInfo.stripeInfo) { + customerId = account.paymentInfo.stripeInfo.customerId; + } else customerId = await this.createCustomer(account); + } + + try { + const params: Stripe.PaymentIntentCreateParams = { + amount: Math.ceil((amount || 1) * 100), + currency: currency || CurrencyEnum.usd, + }; + if (customerId) params.customer = customerId; + if (paymentMethod) params.payment_method = paymentMethod; + + const paymentIntent = await this.stripe.paymentIntents.create(params); + + return paymentIntent; + } catch (error) { + throw newError(6000, { message: error.message }); + } + }; + + public updatePaymentIntent = async ( + paymentIntentId: string, + input: { + amount?: number; + currency?: CurrencyEnum; + paymentMethod?: string; + account?: AccountModel; + }, + ): Promise => { + const { amount, currency, account, paymentMethod } = input; + + let customerId; + if (account) { + // Check if customer exists + if (account.paymentInfo && account.paymentInfo.stripeInfo) { + customerId = account.paymentInfo.stripeInfo.customerId; + } else customerId = await this.createCustomer(account); + } + + try { + const params: Stripe.PaymentIntentUpdateParams | undefined = {}; + // TODO !! + if (amount) params.amount = Math.round(amount * 100); + if (currency) params.currency = currency; + if (customerId) params.customer = customerId; + if (paymentMethod) params.payment_method = paymentMethod; + + if (!_.isEmpty(params)) { + const paymentIntent = await this.stripe.paymentIntents.update(paymentIntentId, params); + return paymentIntent; + } + } catch (error) { + throw newError(6000, { message: error.message }); + } + }; + + public confirmPaymentIntent = async (paymentIntentId: string, input: { paymentMethod: string }): Promise => { + const { paymentMethod } = input; + + try { + const paymentIntent = await this.stripe.paymentIntents.confirm(paymentIntentId, { payment_method: paymentMethod }); + return paymentIntent; + } catch (error) { + throw newError(6000, { message: error.message }); + } + }; + + /* + ██████╗ ██████╗ ██████╗ ██████╗ ██╗ ██╗ ██████╗████████╗ + ██╔══██╗██╔══██╗██╔═══██╗██╔══██╗██║ ██║██╔════╝╚══██╔══╝ + ██████╔╝██████╔╝██║ ██║██║ ██║██║ ██║██║ ██║ + ██╔═══╝ ██╔══██╗██║ ██║██║ ██║██║ ██║██║ ██║ + ██║ ██║ ██║╚██████╔╝██████╔╝╚██████╔╝╚██████╗ ██║ + ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═════╝ ╚═════╝ ╚═════╝ ╚═╝ + + */ + + public createProduct = async (productInfo: { + _id: string; + name: string; + prices: { + amount: number; + currency: 'usd' | 'eur'; + recurring?: { + type: 'licensed' | 'metered'; + interval: 'week' | 'month' | 'year'; + intervalPeriod: number; + }; + }[]; + description?: string; + images?: string[]; + }): Promise<{ sProduct: Stripe.Product; sPrices: Stripe.Price[] }> => { + try { + const result: any = { sProduct: {}, sPrices: [] }; + // Create the product + const productCreate = await this.stripe.products.create({ + name: productInfo.name, + description: productInfo.description || productInfo.name, + active: true, + images: productInfo.images || [], + metadata: { _id: productInfo._id }, + statement_descriptor: productInfo.name, + }); + + result.sProduct = productCreate; + // Link all the prices + const productStripeId = productCreate.id; + + for (let index = 0; index < productInfo.prices.length; index++) { + const price = productInfo.prices[index]; + const priceInfo: Stripe.PriceCreateParams = { + currency: price.currency, + unit_amount: price.amount, + product: productStripeId, + metadata: { productId: productInfo._id }, + }; + + if (price.recurring) { + priceInfo.recurring = { + interval: price.recurring.interval, + interval_count: price.recurring.intervalPeriod, + usage_type: price.recurring.type, + }; + } + + result.sPrices.push(await this.stripe.prices.create(priceInfo)); + } + + return result; + } catch (error) { + throw newError(6000, { message: error.message }); + } + }; + + // public updateProduct = async (paymentMethodId: string): Promise => { + // try { + // // Check if that card exists already + + // const paymentMethod = await this.stripe.paymentMethods.detach(paymentMethodId); + + // return paymentMethod; + // } catch (error) { + // throw newError(6000, { message: error.message }); + // } + // }; + + // public deleteProduct = async (paymentMethodId: string): Promise => { + // try { + // // Check if that card exists already + + // const paymentMethod = await this.stripe.paymentMethods.detach(paymentMethodId); + + // return paymentMethod; + // } catch (error) { + // throw newError(6000, { message: error.message }); + // } + // }; + + /* +████████╗██████╗ █████╗ ███╗ ██╗███████╗███████╗███████╗██████╗ ███████╗ +╚══██╔══╝██╔══██╗██╔══██╗████╗ ██║██╔════╝██╔════╝██╔════╝██╔══██╗██╔════╝ + ██║ ██████╔╝███████║██╔██╗ ██║███████╗█████╗ █████╗ ██████╔╝███████╗ + ██║ ██╔══██╗██╔══██║██║╚██╗██║╚════██║██╔══╝ ██╔══╝ ██╔══██╗╚════██║ + ██║ ██║ ██║██║ ██║██║ ╚████║███████║██║ ███████╗██║ ██║███████║ + ╚═╝ ╚═╝ ╚═╝╚═╝ ╚═╝╚═╝ ╚═══╝╚══════╝╚═╝ ╚══════╝╚═╝ ╚═╝╚══════╝ + + */ + + public reimburse = async (transactionId: string): Promise => { + try { + const stripeRes = await this.stripe.refunds.create({ payment_intent: transactionId }); + if (stripeRes.status == 'succeeded' || stripeRes.status == 'pending') { + return stripeRes; + } else { + throw newError(6000, { message: stripeRes.failure_reason }); + } + } catch (error) { + throw newError(6000, { message: error.message }); + } + }; +} diff --git a/services/module-payments/services/stripe/paymentIntent.helper.ts b/services/module-payments/services/stripe/paymentIntent.helper.ts new file mode 100644 index 0000000..e7c71e1 --- /dev/null +++ b/services/module-payments/services/stripe/paymentIntent.helper.ts @@ -0,0 +1,14 @@ +import { StripePayoutInfo } from '@services/module-payments/components/components.payments'; +import AccountModel from '@src/accounts/account.model'; +import Stripe from 'stripe'; + +export const updateStripePayoutInfo = async (account: AccountModel, stripeCreateAccount: Stripe.Response): Promise => { + const stripePayoutInfo: StripePayoutInfo = { + accountId: stripeCreateAccount.id, + payoutsEnabled: stripeCreateAccount.payouts_enabled, + chargesEnabled: stripeCreateAccount.charges_enabled, + detailsSubmitted: stripeCreateAccount.details_submitted, + }; + + return await account.updateOneCustom({ _id: account._id }, { $set: { 'payoutInfo.stripeInfo': stripePayoutInfo } }, null); +}; diff --git a/services/module-payments/services/stripe/payoutAccounts.helper.ts b/services/module-payments/services/stripe/payoutAccounts.helper.ts new file mode 100644 index 0000000..e69de29 diff --git a/src/__components/components.ts b/src/__components/components.ts new file mode 100644 index 0000000..b7bf308 --- /dev/null +++ b/src/__components/components.ts @@ -0,0 +1,157 @@ +import { registerEnumType, ID } from 'type-graphql'; +import { ObjectType, InputType, Int, Field } from 'type-graphql'; +import { GraphQLJSONObject } from 'graphql-type-json'; +import ImageComponent from '@seed/interfaces/components'; +import { AddressComponent } from '@seed/interfaces/components.geo'; + +/* + ███████╗███╗ ██╗██╗ ██╗███╗ ███╗ + ██╔════╝████╗ ██║██║ ██║████╗ ████║ + █████╗ ██╔██╗ ██║██║ ██║██╔████╔██║ + ██╔══╝ ██║╚██╗██║██║ ██║██║╚██╔╝██║ + ███████╗██║ ╚████║╚██████╔╝██║ ╚═╝ ██║ + ╚══════╝╚═╝ ╚═══╝ ╚═════╝ ╚═╝ ╚═╝ +*/ + +export enum RessourceEnum { + 'workspaces' = 'workspaces', +} +registerEnumType(RessourceEnum, { + name: 'RessourceEnum', +}); + +export enum ListEnum { + 'equipment' = 'equipment', + 'feature' = 'feature', +} +registerEnumType(ListEnum, { + name: 'ListEnum', +}); + +export enum AvailableTranslation { + en = 'en', + fr = 'fr', + nl = 'nl', + de = 'de', +} +registerEnumType(AvailableTranslation, { + name: 'AvailableTranslation', +}); + +export enum WorkspaceTypeEnum { + meeting = 'meeting', + individual = 'individual', + open = 'open', +} +registerEnumType(WorkspaceTypeEnum, { + name: 'WorkspaceTypeEnum', +}); + +export enum ProductTypeEnum { + 'none' = 'none', + 'subscription' = 'subscription', +} +registerEnumType(ProductTypeEnum, { + name: 'ProductTypeEnum', +}); + +export enum CommentModelEnum { + bookings = 'bookings', +} +registerEnumType(CommentModelEnum, { + name: 'CommentModelEnum', +}); + +export enum BookingsRessourceEnum { + workspaces = 'workspaces', +} +registerEnumType(BookingsRessourceEnum, { + name: 'BookingsRessourceEnum', +}); + +export enum AvailableCurrency { + eur = 'eur', + usd = 'usd', +} +registerEnumType(AvailableCurrency, { + name: 'AvailableCurrency', +}); + +export enum ProductRessourceEnum { + 'workspaces' = 'workspaces', +} +registerEnumType(ProductRessourceEnum, { + name: 'ProductRessourceEnum', +}); + +/* + ██████╗ ██████╗ ███╗ ███╗██████╗ ██████╗ ███╗ ██╗███████╗███╗ ██╗████████╗███████╗ + ██╔════╝██╔═══██╗████╗ ████║██╔══██╗██╔═══██╗████╗ ██║██╔════╝████╗ ██║╚══██╔══╝██╔════╝ + ██║ ██║ ██║██╔████╔██║██████╔╝██║ ██║██╔██╗ ██║█████╗ ██╔██╗ ██║ ██║ ███████╗ + ██║ ██║ ██║██║╚██╔╝██║██╔═══╝ ██║ ██║██║╚██╗██║██╔══╝ ██║╚██╗██║ ██║ ╚════██║ + ╚██████╗╚██████╔╝██║ ╚═╝ ██║██║ ╚██████╔╝██║ ╚████║███████╗██║ ╚████║ ██║ ███████║ + ╚═════╝ ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═════╝ ╚═╝ ╚═══╝╚══════╝╚═╝ ╚═══╝ ╚═╝ ╚══════╝ +*/ + +@ObjectType() +@InputType('TranslatableInput') +export class TranslatableComponent { + @Field({ nullable: true }) + en?: string; + @Field() + fr: string; + @Field({ nullable: true }) + nl?: string; + @Field({ nullable: true }) + de?: string; +} + +@ObjectType() +@InputType('BuyerInfo') +export class BuyerInfoComponent { + @Field() + firstName: string; + + @Field() + lastName: string; + + @Field() + email: string; + + @Field({ nullable: true }) + company?: string; + + @Field({ nullable: true }) + vatNumber?: string; + + @Field(() => AddressComponent) + address: AddressComponent; +} + +@ObjectType() +@InputType('MainImage') +export class MainImageComponent { + @Field(() => ImageComponent, { nullable: true }) + main?: ImageComponent; + + @Field(() => ImageComponent, { nullable: true }) + one?: ImageComponent; + + @Field(() => ImageComponent, { nullable: true }) + two?: ImageComponent; + + @Field(() => ImageComponent, { nullable: true }) + three?: ImageComponent; + + @Field(() => ImageComponent, { nullable: true }) + four?: ImageComponent; +} + +@ObjectType() +export class WorkspaceOwnerComponent { + @Field(() => String, { nullable: true }) + firstName?: string; + + @Field(() => String, { nullable: true }) + lastName?: string; +} diff --git a/src/__indexes/__bootstrap.ts b/src/__indexes/__bootstrap.ts new file mode 100644 index 0000000..8e67f64 --- /dev/null +++ b/src/__indexes/__bootstrap.ts @@ -0,0 +1,29 @@ + +import { createApolloContext } from '@seed/graphql/Middleware'; +import { clog } from '@seed/helpers/Utils'; +import { BookingCronService } from '@services/module-booking/functions/crons/booking.cron.service'; +import OrderEngineResolver from '@services/module-payments/functions/orders/resolvers/app/app.resolver'; +import OrderModel from '@src/orders/order.model'; +import { afterOrderCancel, afterPaymentSuccess } from '@src/orders/services/order.service'; + +export const runBeforeAll = async () => { + + // const bservice = new BookingCronService(); + // bservice.removeDrafts(); + + try { + // const orderId = "655ef797-cb09-40e3-927a-8c1d9d668315"; + // afterPaymentSuccess(await new OrderModel().getOne({query:{_id:orderId}})) + + // const oResolver = new OrderEngineResolver(); + // const ctx = await createApolloContext(); + // oResolver.validateOrder(orderId) + return; + + } catch (error) { + + } + + + +}; diff --git a/src/__indexes/__collections.ts b/src/__indexes/__collections.ts new file mode 100644 index 0000000..f546f90 --- /dev/null +++ b/src/__indexes/__collections.ts @@ -0,0 +1,11 @@ +import { registerEnumType } from 'type-graphql'; + +export enum ModelCollectionEnum { + 'streams' = 'stream.changes', + accounts = 'accounts', + workspaces = 'workspaces', + bookings = 'bookings', +} +registerEnumType(ModelCollectionEnum, { + name: 'ModelLoadersEnum', +}); diff --git a/src/__indexes/__loaders.ts b/src/__indexes/__loaders.ts new file mode 100644 index 0000000..85ea416 --- /dev/null +++ b/src/__indexes/__loaders.ts @@ -0,0 +1,68 @@ +import { buildLoader, DataLoader } from '@seed/services/database/LoaderService'; + +import AccountModel from '@src/accounts/account.model'; +import ListModel from '@services/module-cms/functions/lists/list.model'; +import CategoryModel from '@services/module-cms/functions/categories/category.model'; +import CommentModel from '@services/module-comments/functions/comments/comment.model'; +import WorkspaceModel from '@src/workspace/workspace.model'; +import BookingModel from '@services/module-booking/functions/bookings/bookings.engine.model'; +import StreamEngineModel from '@seed/engine/utils/streams/stream.model'; +import PromoModel from '@services/module-payments/functions/promos/promo.model'; +import NotificationModel from '@seed/services/notifications/notifications.model'; +import PageModel from '@services/module-cms/functions/pages/pages.model'; +// import MessageModel from '@services/api-messaging/functions/messages/message.model'; + +export default class Loaders { + /* MODULES */ + + listLoader: DataLoader; + categoryLoader: DataLoader; + + commentLoader: DataLoader; + + promoLoader: DataLoader; + + + /* APP */ + + accountLoader: DataLoader; + workspaceLoader: DataLoader; + + public constructor() { + /* MODULES */ + + this.listLoader = buildLoader(new ListModel()); + this.categoryLoader = buildLoader(new CategoryModel()); + + // this.commentLoader = buildLoader(new CommentModel()); + + this.promoLoader = buildLoader(new PromoModel()); + + + /* APP */ + + this.accountLoader = buildLoader(new AccountModel()); + + this.workspaceLoader = buildLoader(new WorkspaceModel()); + } + + // /* MODULES */ + // public commentRessourceLoader = (args?: GetArgs): DataLoader => { + // return buildOneToManyFilterLoader(new CommentModel(), 'ressourceId', { parentId: { $exists: false } }); + // }; + + // public commentRepliesLoader = (args?: GetArgs): DataLoader => { + // return buildOneToManyLoader(new CommentModel(), 'parentId'); + // }; +} + +export const modelsLoaders = { + 'stream.changes': new StreamEngineModel(), + streams: new StreamEngineModel(), + notifications: new NotificationModel(), + accounts: new AccountModel(), + pages: new PageModel(), + + workspaces: new WorkspaceModel(), + bookings: new BookingModel(), +}; diff --git a/src/__indexes/__resolvers.index.admin.ts b/src/__indexes/__resolvers.index.admin.ts new file mode 100644 index 0000000..5d9cff5 --- /dev/null +++ b/src/__indexes/__resolvers.index.admin.ts @@ -0,0 +1,25 @@ +import 'reflect-metadata'; +import { NonEmptyArray } from 'type-graphql/dist/interfaces/NonEmptyArray'; + +/* + * SEED & MODULES + */ +import AccountBaseResolver from '@services/module-accounts/account.resolver'; +import AccountAdminBaseResolver from '@services/module-accounts/account.resolver.admin'; + +/* + * APP RELATED + */ +import AccountResolver from '@src/accounts/account.resolver'; +import { ApiCMSAdminResolvers } from '@services/module-cms'; + +export const AdminResolvers = [ + /* Modules */ + ...ApiCMSAdminResolvers, + + AccountAdminBaseResolver, + AccountBaseResolver, + + /* App specific */ + AccountResolver, +]; diff --git a/src/__indexes/__resolvers.index.ts b/src/__indexes/__resolvers.index.ts new file mode 100644 index 0000000..de7c49f --- /dev/null +++ b/src/__indexes/__resolvers.index.ts @@ -0,0 +1,50 @@ +import 'reflect-metadata'; + +/* + * SEED & MODULES + */ +import AccountBaseResolver from '@services/module-accounts/account.resolver'; + +/* + * APP RELATED + */ +import AccountResolver from '@src/accounts/account.resolver'; +import GooglePlaceResolver from '@seed/services/geolocation/functions/GooglePlaceResolver'; +import { ApiCMSAppResolvers } from '@services/module-cms'; +import { PaymentAppEngineResolvers, PayoutAppEngineResolvers } from '@services/module-payments'; +import { AccountFavResolver } from '@services/module-favorites/functions/favLikes/favLikes.resolver'; + +import WorkspaceResolver from '@src/workspace/workspace.resolver'; +import { BookingEngineResolvers } from '@services/module-booking/functions/bookings/resolvers/bookings.engine.resolver'; +import CheckoutResolver, { OrderFieldResolver } from '@src/orders/order.resolver'; +import OrderEngineResolver from '@services/module-payments/functions/orders/resolvers/app/app.resolver'; +import { BookingResolvers } from '@src/bookings/resolvers/bookings.resolver'; +import WorkspaceFieldResolver from '@src/workspace/workspace.resolver.fields'; +import { ApiMessagingResolver } from '@services/api-messaging/handler'; + +export const AppResolvers = [ + /* Modules */ + ...ApiCMSAppResolvers, + ...ApiMessagingResolver, + + AccountBaseResolver, + GooglePlaceResolver, + + /* App specific */ + AccountResolver, + AccountFavResolver, + + WorkspaceResolver, + WorkspaceFieldResolver, + + ...BookingEngineResolvers, + ...BookingResolvers, + + OrderEngineResolver, + OrderFieldResolver, + + ...PaymentAppEngineResolvers, + ...PayoutAppEngineResolvers, + + CheckoutResolver, +]; diff --git a/src/accounts/account.components.ts b/src/accounts/account.components.ts new file mode 100644 index 0000000..9592e03 --- /dev/null +++ b/src/accounts/account.components.ts @@ -0,0 +1,152 @@ +import { AddressComponent } from '@seed/interfaces/components.geo'; +import { Field, InputType, ObjectType, registerEnumType } from 'type-graphql'; + +/* + ███████╗███╗ ██╗██╗ ██╗███╗ ███╗ + ██╔════╝████╗ ██║██║ ██║████╗ ████║ + █████╗ ██╔██╗ ██║██║ ██║██╔████╔██║ + ██╔══╝ ██║╚██╗██║██║ ██║██║╚██╔╝██║ + ███████╗██║ ╚████║╚██████╔╝██║ ╚═╝ ██║ + ╚══════╝╚═╝ ╚═══╝ ╚═════╝ ╚═╝ ╚═╝ +*/ + +export enum AccountTypeEnum { + admin = 'admin', + user = 'user', + pro = 'pro', + public = 'public', + organisationOwner = 'organisationOwner', + storeManager = 'storeManager', +} +registerEnumType(AccountTypeEnum, { + name: 'AccountTypeEnum', +}); + +export enum PersonaEnum { + independant = 'independant', + ceo = 'ceo', + employee = 'employee', + association = 'association', + student = 'student', +} +registerEnumType(PersonaEnum, { + name: 'PersonaEnum', +}); + +export enum SocietyType { + independant = 'independant', + company = 'company', +} +registerEnumType(SocietyType, { + name: 'SocietyType', +}); + +export enum TransferType { + bankAccount = 'bankAccount', +} +registerEnumType(TransferType, { + name: 'TransferType', +}); + +export enum SettingsTypeEnum { + notifications = 'notifications', + promotions = 'promotions', + reminders = 'reminders', +} +registerEnumType(SettingsTypeEnum, { + name: 'SettingsTypeEnum', +}); + +@ObjectType() +@InputType('IndependantInfoComponentInput') +export class IndependantInfoComponent { + @Field() + companyNumber: string; +} + +@ObjectType() +@InputType('CeoInfoComponentInput') +export class CeoInfoComponent { + @Field() + companyNumber: string; + + @Field({ nullable: true }) + companyName?: string; + + @Field({ nullable: true }) + fiscalForm?: string; +} + +@ObjectType() +@InputType('EmployeeInfoComponentInput') +export class EmployeeInfoComponent { + @Field() + companyNumber: string; + + @Field({ nullable: true }) + companyName?: string; + + @Field({ nullable: true }) + fiscalForm?: string; + + @Field({ nullable: true }) + jobFunction?: string; +} + +@ObjectType() +@InputType('AssociationInfoComponentInput') +export class AssociationInfoComponent { + @Field() + companyNumber: string; + + @Field({ nullable: true }) + companyName?: string; + + @Field({ nullable: true }) + fiscalForm?: string; + + @Field({ nullable: true }) + jobFunction?: string; +} + +@ObjectType() +@InputType('StudentInfoComponentInput') +export class StudentInfoComponent { + @Field({ nullable: true }) + schoolName?: string; + + @Field({ nullable: true }) + sectionName?: string; + + @Field({ nullable: true }) + schoolYear?: string; +} + +@ObjectType() +@InputType('InvoiceProfileInput') +export class InvoiceProfile { + @Field(() => SocietyType) + type: SocietyType; + + @Field() + companyNumber: string; + + @Field({ nullable: true }) + companyName?: string; + + @Field({ nullable: true }) + fiscalForm?: string; + + @Field(() => AddressComponent, { nullable: true }) + address: AddressComponent; +} + +@ObjectType() +@InputType('TransferProfileInput') +export class TransferProfile { + @Field(() => TransferType) + type: TransferType; + + @Field() + accountNumber: string; +} diff --git a/src/accounts/account.helper.ts b/src/accounts/account.helper.ts new file mode 100644 index 0000000..879b0d1 --- /dev/null +++ b/src/accounts/account.helper.ts @@ -0,0 +1,3 @@ +import AccountModel from './account.model'; + +export const checkProfileCompletion = (model: AccountModel) => {}; diff --git a/src/accounts/account.model.ts b/src/accounts/account.model.ts new file mode 100644 index 0000000..b227198 --- /dev/null +++ b/src/accounts/account.model.ts @@ -0,0 +1,77 @@ +import { ObjectType, Field, ArgsType, InputType } from 'type-graphql'; +import { Equals } from 'class-validator'; + +import BaseAccountModel, { EditBaseAccountInput, NewBaseAccountInput } from '@services/module-accounts/account.model'; +import { Permission } from '@seed/interfaces/permission'; + +import { GetArgs } from '@seed/graphql/Request'; +import { AccountTypeEnum } from './account.components'; +import { FavoriteComponent } from '@services/module-favorites/components'; +import { AccountPaymentInfo, AccountPayoutInfo } from '@services/module-payments/components/components.payments'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.admin], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class AccountModel extends BaseAccountModel { + public constructor() { + super(permissions); + } + + @Field(() => AccountPaymentInfo, { nullable: true }) + paymentInfo?: AccountPaymentInfo; + + @Field(() => AccountPayoutInfo, { nullable: true }) + payoutInfo?: AccountPayoutInfo; + + @Field() + terms: boolean; + + @Field(() => [FavoriteComponent], { nullable: 'itemsAndList' }) + favLikes?: FavoriteComponent[]; + + // @Field(() => PersonaEnum, { nullable: true }) + // currentPersona?: PersonaEnum; + + // @Field(() => IndependantInfoComponent, { nullable: true }) + // independantInfo?: IndependantInfoComponent; + + // @Field(() => CeoInfoComponent, { nullable: true }) + // ceoInfo?: CeoInfoComponent; + + // @Field(() => EmployeeInfoComponent, { nullable: true }) + // employeeInfo?: EmployeeInfoComponent; + + // @Field(() => AssociationInfoComponent, { nullable: true }) + // associationInfo?: AssociationInfoComponent; + + // @Field(() => StudentInfoComponent, { nullable: true }) + // studentInfo?: StudentInfoComponent; + + searchOptions(): string[] { + return []; + } + async afterUpdate(): Promise { + // Check if current profile is complete + } +} + +@ArgsType() +export class AccountArgs { + @Field(() => GetArgs, { nullable: true }) + pagination?: GetArgs; +} + +@InputType() +export class NewAccountInput extends NewBaseAccountInput implements Partial { + @Field() + @Equals(true) + terms: boolean; +} + +@InputType() +export class EditAccountInput extends EditBaseAccountInput implements Partial {} diff --git a/src/accounts/account.resolver.ts b/src/accounts/account.resolver.ts new file mode 100644 index 0000000..e44ba77 --- /dev/null +++ b/src/accounts/account.resolver.ts @@ -0,0 +1,57 @@ +import { Resolver, Ctx, FieldResolver, Root, Mutation, Arg, Authorized } from 'type-graphql'; +import { ApolloContext } from '@seed/interfaces/context'; + +import AccountModel, { NewAccountInput, EditAccountInput } from '@src/accounts/account.model'; +import { signinGeneric } from '@services/module-accounts/services/AccountService'; +import { AccountTypeEnum } from './account.components'; +import { editOneGeneric } from '@seed/graphql/BaseService'; +import AccountBaseResolver from '@services/module-accounts/account.resolver'; +import { sendNotification } from '@seed/services/notifications/services/NotificationService'; +import { NotificationEnum } from '@config/config'; +import { AvailableTranslation } from '@src/__components/components'; + +@Resolver(AccountModel) +export default class AccountResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ + + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + + */ + + @Mutation(() => AccountModel) + async register(@Arg('input') input: NewAccountInput, @Ctx() ctx: ApolloContext): Promise { + const model = new AccountModel(); + const newAccount = await signinGeneric(model, AccountTypeEnum.user, input, ctx); + + await sendNotification({ + key: NotificationEnum.welcomeUser, + emails: [newAccount.email], + language: newAccount.language || AvailableTranslation.fr, + ressource: {...newAccount, siteLink: process.env.SITE_URL || 'https://work-in-flex.com'}, + }) + + return newAccount as any; + } +} diff --git a/src/accounts/account.service.ts b/src/accounts/account.service.ts new file mode 100644 index 0000000..e69de29 diff --git a/src/bookings/resolvers/bookings.resolver.ts b/src/bookings/resolvers/bookings.resolver.ts new file mode 100644 index 0000000..86c32b7 --- /dev/null +++ b/src/bookings/resolvers/bookings.resolver.ts @@ -0,0 +1,140 @@ +import { EngineMiddleware } from '@seed/graphql/MiddlewareV2'; +import { StatusEnum } from '@seed/interfaces/components'; +import { ApolloContext } from '@seed/interfaces/context'; +import RoomModel from '@services/api-messaging/functions/rooms/room.model'; +import { roomsGetOneByAccountsIds } from '@services/api-messaging/functions/rooms/room.service'; +import { BookingService } from '@services/module-booking/functions/bookings/booking.service'; +import BookingModel from '@services/module-booking/functions/bookings/bookings.engine.model'; +import { BookingEngineMutationResolver } from '@services/module-booking/functions/bookings/resolvers/bookings.engine.resolver'; +import { BookingEngineNewInputSchema } from '@services/module-booking/functions/bookings/schemas/bookings.engine.input'; +import { BookingEngineSchema } from '@services/module-booking/functions/bookings/schemas/bookings.engine.schema'; +import { DurationTypeEnum } from '@services/module-booking/functions/components'; +import { formatSlot, formatStartToEnd, SlotFormat, StartToEndFormat } from '@services/module-booking/functions/helper'; +import { OrderEngineSchema } from '@services/module-payments/functions/orders/schemas/order.schema'; +import { beginCheckoutWithOrder, finaliseCheckoutWithOrder } from '@services/module-payments/functions/orders/services/OrderService'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import WorkspaceModel from '@src/workspace/workspace.model'; +import { BookingsRessourceEnum } from '@src/__components/components'; +import { ModelCollectionEnum } from '@src/__indexes/__collections'; +import _ from 'lodash'; +import { Arg, Authorized, Ctx, FieldResolver, Mutation, Resolver, Root } from 'type-graphql'; +import { BookingNewInputSchema } from '../schemas/bookings.input'; +import { BookingSchema } from '../schemas/bookings.schema'; + +@Resolver(BookingEngineSchema) +export class BookingFieldResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ + + @FieldResolver(() => WorkspaceModel, { nullable: true }) + async workspace(@Root() booking: BookingSchema, @Ctx() ctx: ApolloContext): Promise { + // Get the correct booking model + const ref = _.find(booking.paths, { ressourceModel: BookingsRessourceEnum.workspaces }); + if (ref) return ctx.ctx.loaders.workspaceLoader.load(ref.ressourceId); + return; + } + + @FieldResolver(() => RoomModel, { nullable: true }) + async room(@Root() booking: BookingSchema, @Ctx() ctx: ApolloContext): Promise { + const workspace = await this.workspace(booking, ctx); + if (workspace && workspace.createdBy) { + const workspaceOwner = workspace.createdBy; + const bookingUser = booking.ownerId; + + return roomsGetOneByAccountsIds([workspaceOwner, bookingUser], { ctx }); + } + return; + } +} + +@Resolver(BookingSchema) +export class BookingQueryResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ +} + +@Resolver(BookingSchema) +export class BookingMutationResolver { + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ + + @Mutation(() => OrderEngineSchema) + @EngineMiddleware({ + authorization: [AccountTypeEnum.user], + validations: [{ schema: BookingEngineNewInputSchema }], + }) + async reserveOneBooking(@Arg('input') input: BookingNewInputSchema, @Ctx() ctx: ApolloContext): Promise { + const zone = process.env.TIMEZONE || 'utc'; + + const workspace = await new WorkspaceModel().getOne({ _id: input.ressourceId }, ctx); + + const removeWeekend = workspace.availability.noWeekend || false; + + let formatted: StartToEndFormat | SlotFormat; + + if (input.durationType == DurationTypeEnum.startToEnd) formatted = formatStartToEnd(input.startToEnd!, { zone }); + else formatted = formatSlot(input.slot!, { zone, removeWeekend }); + + // Create the booking + // Remove all the previous draft + await (await new BookingModel().db()).deleteMany({ ownerId: ctx.ctx.user._id, status: StatusEnum.draft }); + const data = await new BookingService().createOneBookingWithModel({ + model: workspace, + input, + status: StatusEnum.draft, + availabilities: workspace.availability, + ctx, + }); + + // Get the quantityHours + const quantityHours = formatted.bookingDurationInMinutes / 60; + // TODO : Add taxes + + return await beginCheckoutWithOrder({ + lines: [workspace.getLineProduct({ quantity: quantityHours })], + contactInfo: input.checkoutInfo, + input, + paths: [{'ressourceModel':ModelCollectionEnum.bookings,ressourceId:data._id},{'ressourceModel':ModelCollectionEnum.workspaces,ressourceId:workspace._id}], + ctx, + }); + + // Send the notifications + // const promises = [ + // sendNotification({ + // key: 'agendaUserConfirm', + // emails: [ctx.ctx.user.email], + // language: ctx.ctx.user.language, + // ressource: { formatted, pro, user, ...data }, + // }), + // sendNotification({ + // key: 'agendaProConfirm', + // emails: [pro.email || pro.googleEmail], + // language: ctx.ctx.user.language, + // ressource: { formatted, pro, user, ...data }, + // }), + // ]; + // await promiseAll(promises); + + // return data; + } +} + +export const BookingResolvers = [BookingFieldResolver, BookingQueryResolver, BookingMutationResolver]; diff --git a/src/bookings/resolvers/getEventDescription.ts b/src/bookings/resolvers/getEventDescription.ts new file mode 100644 index 0000000..1048866 --- /dev/null +++ b/src/bookings/resolvers/getEventDescription.ts @@ -0,0 +1,18 @@ +import { interpolate } from '@seed/helpers/Utils'; +import EmailSettingsModel from '@services/module-cms/functions/emails-settings/emails-settings.model'; +import { AvailableTranslation } from '@src/__components/components'; + +export const getEventDescription = async (input: any, language: AvailableTranslation = AvailableTranslation.fr): Promise => { + const emailSettings = new EmailSettingsModel(); + try { + await emailSettings.getOne({ key: 'agendaProConfirm' }, null); + } catch (error) { + console.error('NOTIFICATION.ERROR', { message: 'notfound', key: 'agendaProConfirm' }); + return ''; + } + + const body = emailSettings.body && emailSettings.body[language] ? emailSettings.body[language] : ''; + const string = interpolate(body || '', input); + + return string; +}; diff --git a/src/bookings/schemas/bookings.input.ts b/src/bookings/schemas/bookings.input.ts new file mode 100644 index 0000000..1bcd09e --- /dev/null +++ b/src/bookings/schemas/bookings.input.ts @@ -0,0 +1,20 @@ +import { Field, InputType, Int, ArgsType, ObjectType } from 'type-graphql'; + +import { BookingEngineArgsSchema, BookingEngineNewInputSchema } from '@services/module-booking/functions/bookings/schemas/bookings.engine.input'; +import { CheckoutEngineInput } from '@services/module-payments/functions/orders/schemas/order.schema.input'; + +@ObjectType('BookingNewInput') +@InputType() +export class BookingNewInputSchema extends BookingEngineNewInputSchema { + @Field(() => Int, { nullable: true }) + capacity: number; + + @Field({ nullable: true }) + comments?: string; + + @Field(() => CheckoutEngineInput, { nullable: true }) + checkoutInfo?: CheckoutEngineInput; +} + +@ArgsType() +export class BookingArgsSchema extends BookingEngineArgsSchema {} diff --git a/src/bookings/schemas/bookings.schema.ts b/src/bookings/schemas/bookings.schema.ts new file mode 100644 index 0000000..34bfbe0 --- /dev/null +++ b/src/bookings/schemas/bookings.schema.ts @@ -0,0 +1,22 @@ +import { IEngineSchema, MetaBy, MetaPermissions } from '@seed/engine/EngineSchema'; +import { BookingEngineDBInterfaceSchema } from '@services/module-booking/functions/bookings/schemas/bookings.engine.schema'; +import { Field, Int, ObjectType } from 'type-graphql'; + +@ObjectType() +export class BookingDBInterfaceSchema extends BookingEngineDBInterfaceSchema { + @Field(() => Int) + capacity: number; +} + +@ObjectType() +export class BookingDBSchema extends BookingDBInterfaceSchema {} + +@ObjectType({ implements: IEngineSchema }) +export class BookingSchema extends BookingDBSchema implements IEngineSchema { + _id: string; + organisationId?: string | undefined; + by?: MetaBy | undefined; + permissions: MetaPermissions; + createdAt: Date; + updatedAt: Date; +} diff --git a/src/orders/components/checkout.schema.ts b/src/orders/components/checkout.schema.ts new file mode 100644 index 0000000..27e3b10 --- /dev/null +++ b/src/orders/components/checkout.schema.ts @@ -0,0 +1,6 @@ +import { ObjectType, Field, Ctx, InputType } from 'type-graphql'; + +import { CheckoutEngineInput } from '@services/module-payments/functions/orders/schemas/order.schema.input'; + +@InputType() +export class CheckoutInput extends CheckoutEngineInput {} diff --git a/src/orders/components/components.cart.ts b/src/orders/components/components.cart.ts new file mode 100644 index 0000000..610e5d1 --- /dev/null +++ b/src/orders/components/components.cart.ts @@ -0,0 +1,10 @@ +import { OrderEngineNewSchema } from '@services/module-payments/functions/orders/schemas/order.schema.input'; +import { ObjectType, InputType } from 'type-graphql'; + +@ObjectType() +@InputType('OrderNewInput') +export class OrderNewInputSchema extends OrderEngineNewSchema {} + +@ObjectType() +@InputType('CartLineOptionInputComponent') +export class CartLineOptionComponent {} diff --git a/src/orders/components/line.schema.ts b/src/orders/components/line.schema.ts new file mode 100644 index 0000000..db0f3bc --- /dev/null +++ b/src/orders/components/line.schema.ts @@ -0,0 +1,44 @@ +import { ObjectType, Field, Ctx, InputType } from 'type-graphql'; + +import { ApolloContext, ApolloContextLoadersOnly } from '@seed/interfaces/context'; +import { LineItemGenericSchema } from '@services/module-payments/functions/orders/carts/schemas/line.schema'; +import { ProductRessourceEnum } from '@src/__components/components'; +import WorkspaceModel from '@src/workspace/workspace.model'; + +import { createUnionType } from 'type-graphql'; +import { plainToClass } from 'class-transformer'; +import { linePrice, lineVatPrice } from '../services/line.service'; + +export const ProductUnion = createUnionType({ + name: 'ProductUnion', // the name of the GraphQL union + types: () => [WorkspaceModel] as const, // function that returns tuple of object types classes +}); + +@InputType('LineItemSchemaInput') +@ObjectType() +export class LineItemBaseSchema extends LineItemGenericSchema { + @Field(() => ProductUnion, { nullable: true }) + async getProduct(@Ctx() ctx: ApolloContext): Promise { + if (this.productRessource == ProductRessourceEnum.workspaces) { + return plainToClass(WorkspaceModel, await ctx.ctx.loaders.workspaceLoader.load(this.productId)); + } + + return; + } + + @Field(() => Number) + async getLinePrice(@Ctx() ctx: ApolloContext): Promise { + return await linePrice(this, ctx); + } + + @Field(() => Number) + async getLineVatPrice(@Ctx() ctx: ApolloContext): Promise { + return await lineVatPrice(this, ctx); + } +} + +@ObjectType() +export class LineItemDBSchema extends LineItemBaseSchema {} + +@ObjectType() +export class LineItemSchema extends LineItemBaseSchema {} diff --git a/src/orders/order.model.ts b/src/orders/order.model.ts new file mode 100644 index 0000000..3d0e617 --- /dev/null +++ b/src/orders/order.model.ts @@ -0,0 +1,23 @@ +import { IEngineSchema } from '@seed/engine/EngineSchema'; +import OrderEngineModel from '@services/module-payments/functions/orders/order.model'; +import { OrderEngineDBInterfaceSchema, OrderEngineSchema } from '@services/module-payments/functions/orders/schemas/order.schema'; +import { ObjectType, Field, ArgsType } from 'type-graphql'; + +@ObjectType() +export default class OrderModel extends OrderEngineModel { + public constructor(input?: OrderEngineDBInterfaceSchema & Partial) { + super(input); + } + + searchOptions(): string[] { + return []; + } + + async afterPaymentSuccess(order:OrderEngineSchema): Promise { + console.log('ok', this._id); + } + + async afterPaymentFailed(order:OrderEngineSchema): Promise { + console.log('ok', this._id); + } +} diff --git a/src/orders/order.resolver.admin.ts b/src/orders/order.resolver.admin.ts new file mode 100644 index 0000000..e69de29 diff --git a/src/orders/order.resolver.ts b/src/orders/order.resolver.ts new file mode 100644 index 0000000..4dee971 --- /dev/null +++ b/src/orders/order.resolver.ts @@ -0,0 +1,55 @@ +import { Resolver, Authorized, Ctx, FieldResolver, Root } from 'type-graphql'; +import { Mutation, Arg } from 'type-graphql'; +import { ApolloContext } from '@seed/interfaces/context'; + +// From Module + +// From SRC +import OrderModel from '@src/orders/order.model'; +import CheckoutEngineResolver from '@services/module-payments/functions/orders/resolvers/app/checkout.resolver'; +import { OrderEngineSchema } from '@services/module-payments/functions/orders/schemas/order.schema'; +import RoomModel from '@services/api-messaging/functions/rooms/room.model'; +import { roomsGetOneByAccountsIds } from '@services/api-messaging/functions/rooms/room.service'; +import { BookingSchema } from '@src/bookings/schemas/bookings.schema'; +import WorkspaceModel from '@src/workspace/workspace.model'; +import { BookingsRessourceEnum } from '@src/__components/components'; +import _ from 'lodash'; +import { BookingNewInputSchema } from '@src/bookings/schemas/bookings.input'; + +@Resolver(OrderEngineSchema) +export class OrderFieldResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ + + @FieldResolver(() => BookingNewInputSchema, { nullable: true }) + async getBookingInput(@Root() order: OrderEngineSchema, @Ctx() ctx: ApolloContext): Promise { + if (order.inputData) return order.inputData; + return; + } +} + +@Resolver(OrderModel) +export default class CheckoutResolver extends CheckoutEngineResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ +} diff --git a/src/orders/services/line.service.ts b/src/orders/services/line.service.ts new file mode 100644 index 0000000..47b3594 --- /dev/null +++ b/src/orders/services/line.service.ts @@ -0,0 +1,56 @@ +import { ApolloContext } from '@seed/interfaces/context'; + +import VatModel from '@services/module-payments/functions/vat/vat.model'; +import PromoModel from '@services/module-payments/functions/promos/promo.model'; +import { LineItemDBSchema, ProductUnion } from '@src/orders/components/line.schema'; + +const getProductPrice = async (product: typeof ProductUnion | undefined, quantity: number, ctx: ApolloContext, promoId?: string): Promise => { + let price = product ? (await product.finalPrice()) * quantity : 0; + if (promoId) { + const promo = await new PromoModel().getOne({ _id: promoId }, ctx); + price -= (price * promo.value) / 100; + } + return price; +}; + +const getProductVatPrice = async ( + product: typeof ProductUnion | undefined, + quantity: number, + countryCode: string | undefined, + ctx: ApolloContext, + promoId?: string, +): Promise => { + let vatPrice = 0; + + let vatClass: VatModel; + + if ((product as any).vatClassId) + vatClass = await new VatModel().getOne({ _id: (product as any).vatClassId }, ctx); + else vatClass = await new VatModel().getOne({ 'title': 'standard' }, ctx); + + if (!countryCode) countryCode = process.env.PAYMENT_DEFAULT_COUNTRY || 'BE'; + + const vatValue = vatClass.values[countryCode]; + vatPrice += ((await getProductPrice(product, quantity, ctx)) * vatValue) / 100; + + + if (promoId) { + const promo = await new PromoModel().getOne({ _id: promoId }, ctx); + vatPrice -= (vatPrice * promo.value) / 100; + } + return vatPrice; +}; + +export const linePrice = async (line: LineItemDBSchema, ctx: ApolloContext): Promise => { + let price = 0; + price += await getProductPrice(await line.getProduct(ctx), line.quantity, ctx, line.promoId); + + return price; +}; + +export const lineVatPrice = async (line: LineItemDBSchema, ctx: ApolloContext): Promise => { + let vatPrice = 0; + vatPrice += await getProductVatPrice(await line.getProduct(ctx), line.quantity, line.localeInfo?.countryCode, ctx); + + return vatPrice; +}; diff --git a/src/orders/services/order.service.ts b/src/orders/services/order.service.ts new file mode 100644 index 0000000..612f1d1 --- /dev/null +++ b/src/orders/services/order.service.ts @@ -0,0 +1,186 @@ +import _ from "lodash"; +import { promiseAll } from "@seed/helpers/Utils"; + +import { OrderEngineSchema } from "@services/module-payments/functions/orders/schemas/order.schema"; +import { StatusEnum } from "@seed/interfaces/components"; + +import { ModelCollectionEnum } from "@src/__indexes/__collections"; +import AccountModel from "@src/accounts/account.model"; +import BookingModel from "@services/module-booking/functions/bookings/bookings.engine.model"; +import { sendNotification } from "@seed/services/notifications/services/NotificationService"; +import { NotificationEnum } from "@config/config"; + +import moment from "moment-timezone"; +import { BookingSchema } from "@src/bookings/schemas/bookings.schema"; +import WorkspaceModel from "@src/workspace/workspace.model"; + +//get booking & workspace path +export const getPaths = (order: OrderEngineSchema) => { + const paths = { + bookingPath: _.find(order.paths,{'ressourceModel':ModelCollectionEnum.bookings}), + workspacePath: _.find(order.paths,{'ressourceModel':ModelCollectionEnum.workspaces}), + } + + if (!paths.bookingPath || !paths.workspacePath) { + console.error('SYSTEM ERROR', 'NO BOOKING IN ORDERS') + return + } + + return paths; +} + +//get workspace of order +export const getWorkspace = async(workspacePath): Promise => { + return await new WorkspaceModel().getOne({_id: workspacePath?.ressourceId}, null); +} + +//get owner +export const getOwner = async(ownerId): Promise => { + return await new AccountModel().getOne({_id: ownerId}, null); +} + +//transform slots to string to pass to email +export const transformSlotsToString = (booking: BookingSchema) => { + let slots = ''; + booking.dates.map((item) => { + slots += `
  • ${moment.tz(item.startDate, "Europe/Brussels").format('DD/MM/YYYY')}, ${booking.slot?.startTime} - ${booking.slot?.endTime}
  • `; + }); + + return slots; +} + +//get booking +export const getBooking = async(bookingPath): Promise => { + return await new BookingModel().getOne({ + query:{'_id': bookingPath?.ressourceId}} + ); +} + +export const afterPaymentSuccess = async (order: OrderEngineSchema) => { + const promises: Promise[] = []; + + + const accountModel = new AccountModel(); + const client = await accountModel.getOne({ _id: order.accountId }, null); + + /* + * (1) Make the booking in validated + */ + const paths = getPaths(order); + + const booking = await getBooking(paths?.bookingPath); + + await new BookingModel().updateOne({ + query:{'_id': paths?.bookingPath?.ressourceId},newData: { + status:StatusEnum.validated + }} + ); + + + /* + * (2) Create Invoice + */ + + //line.getPrice doesnt work online for some reason + // await Promise.all( + // order.lines.map(async (line) => { + // return await StripeService.getInstance().createInvoiceItem( + // account.paymentInfo?.stripeInfo?.customerId || '', + // line.price, + // // (await line.getLinePrice(ctx)) + (await line.getLineVatPrice(ctx)), + // line.title, + // order.currency, + // ['txr_1IVZECCvwkniRa7bJi8ZiLhN'], + // ); + // }), + // ); + + // const invoice = await StripeService.getInstance().createInvoice(account.paymentInfo?.stripeInfo?.customerId || '', 'test'); + // await StripeService.getInstance().payInvoice(invoice.id); + + + /* + * (3) Send email + */ + //get workspace + const workspace = await getWorkspace(paths?.workspacePath); + + //get owner + const owner = await getOwner(workspace.createdBy); + + //format dates & slots to string to pass to email + const slots = transformSlotsToString(booking); + + //shortId to display in email + const shortId = booking._id.substr(booking._id.length - 5, 5); + + //send confirm of booking to owner + promises.push( + sendNotification({ + key: NotificationEnum.bookingProConfirm, + emails: [owner.email], + language: owner.language, + ressource: {contact: client.email, shortId, bookingLink: `${process.env.BOOKING_URL}${workspace._id}`, workspace, slots} + }) + ) + //send confirm of booking to client + promises.push( + sendNotification({ + key: NotificationEnum.bookingUserConfirm, + emails: [client.email], + language: client.language, + ressource: {contact: owner.email, shortId, bookingLink: `${process.env.BOOKING_URL}${workspace._id}`, workspace, slots } + }) + ) + + + await promiseAll(promises); +} + +export const afterOrderCancel = async (order: OrderEngineSchema) => { + const promises: Promise[] = []; + + const client = await new AccountModel().getOne({ _id: order.accountId }, null); + + const paths = getPaths(order); + + //get booking + const booking = await getBooking(paths?.bookingPath); + + //get workspace + const workspace = await getWorkspace(paths?.workspacePath); + + //get owner + const owner = await getOwner(workspace.createdBy); + + //format dates & slots to string to pass to email + const slots = transformSlotsToString(booking); + + //shortId to display in email + const shortId = booking._id.substr(booking._id.length - 5, 5); + + //send confirm cancellation of booking to client + promises.push( + sendNotification({ + key: NotificationEnum.bookingUserCancel, + emails: [client.email], + language: client.language, + ressource: {shortId, link: `${process.env.SITE_URL}`, workspace, slots } + }) + ) + + //send confirm cancellation of booking to owner + promises.push( + sendNotification({ + key: NotificationEnum.bookingProCancel, + emails: [owner.email], + language: owner.language, + ressource: {shortId, workspace, slots} + }) +) + + await promiseAll(promises); +} + +export const afterPaymentFailed = async (order: OrderEngineSchema) => { +} \ No newline at end of file diff --git a/src/workspace/workspace.model.ts b/src/workspace/workspace.model.ts new file mode 100644 index 0000000..cb41ae5 --- /dev/null +++ b/src/workspace/workspace.model.ts @@ -0,0 +1,288 @@ +import { ObjectType, Field, ArgsType, InputType, Int } from 'type-graphql'; + +import { Permission } from '@seed/interfaces/permission'; +// import CommentModel from '@services/module-comments/functions/comments/comment.model'; +import { AvailableCurrency, MainImageComponent, ProductRessourceEnum, WorkspaceTypeEnum } from '@src/__components/components'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import { CustomSearchEngine } from '@seed/services/database/DBRequestService'; +import BaseGeoSimpleSEOModel, { + NewGeoBaseSEOSimpleInput, + EditGeoBaseSEOSimpleInput, + BaseGeoArgs, + NewGeoWithAddressBaseSEOSimpleInput, + EditGeoWithAddressBaseSEOSimpleInput, +} from '@seed/services/geolocation/models/BaseGeoSimpleModel'; +import { RangeType } from '@seed/interfaces/components'; +import { AvailabilityComponent } from '@seed/interfaces/components.dates'; +import { + LineProductBaseInterface, + LineProductFunctionInterface, + LineProductInterface, +} from '@services/module-payments/functions/orders/carts/schemas/line.schema'; + +import { v4 as uuid } from 'uuid'; +import { EditBaseSEOSimpleInput, NewBaseSEOSimpleInput } from '@seed/graphql/baseModels/BaseSeoSimpleModel'; +import { AddressComponent, AddressStrictComponent } from '@seed/interfaces/components.geo'; +import { getGeoLocFromAddress, getPlaceFromAddress } from '@seed/services/geolocation/services/GooglePlaceService'; + +const permissions: Permission = { + c: [AccountTypeEnum.admin], + r: [AccountTypeEnum.public], + w: [AccountTypeEnum.admin], + d: [AccountTypeEnum.admin], +}; + +@ObjectType() +export default class WorkspaceModel extends BaseGeoSimpleSEOModel implements LineProductBaseInterface { + public constructor(collectionName?: string, perm?: Permission) { + const cName = collectionName ? collectionName : 'workspaces'; + const permission = perm ? perm : permissions; + super(cName, permission); + } + + @Field(() => WorkspaceTypeEnum) + workspaceType: WorkspaceTypeEnum; + + @Field(() => Int) + pricingPerHour: number; + + @Field(() => AvailableCurrency) + currency: AvailableCurrency; + + @Field(() => Int) + maxCapacity: number; + + @Field(() => Int) + minStay: number; + @Field(() => Int, { nullable: true }) + maxStay?: number; + + @Field(() => AvailabilityComponent) + availability: AvailabilityComponent; + + @Field(() => MainImageComponent) + mainImages: MainImageComponent; + + @Field(() => [String]) + equipmentIds: string[]; + @Field(() => [String]) + featureIds: string[]; + + + @Field(() => AddressComponent) + address(): AddressComponent { + return this.place.address; + } + + @Field() + finalPrice(): number { + // if (this.currentPromo) { + // if (this.currentPromo.promoType == PromoType.percentage) return this.pricingPerHour * (1 - this.currentPromo.value / 100); + // else return this.pricingPerHour - this.currentPromo.value; + // } + return this.pricingPerHour; + } + + /* + * PROMOS + */ + // @Field(() => CurrentPromo, { nullable: true }) + // currentPromo?: CurrentPromo; + + /* + * FIELDS RESOLVERS + */ + + // @Field(() => [CommentModel], { nullable: true }) + // comments?: CommentModel[]; + + searchOptions(): string[] { + return ['workspaceType']; + } + + filterOptions(): string[] { + return ['workspaceType']; + } + + // eslint-disable-next-line @typescript-eslint/no-empty-function + async onCreate(input:NewWorkspaceInput): Promise { + input.place = await getPlaceFromAddress(input.address) + } + + async onUpdate(input:EditWorkspaceInput): Promise { + if (input.address) input.place = await getPlaceFromAddress(input.address) + } + + getLineProduct(input: { quantity: number }): LineProductInterface { + return { + _id: uuid(), + productRessource: ProductRessourceEnum.workspaces, + productId: this._id, + + title: this.title || 'Workspace - ' + this._id, + + quantity: input.quantity, + price: this.pricingPerHour, + salesPrice: this.pricingPerHour, + + finalPrice: this.finalPrice, + }; + } + + getLineProductFunctions(): LineProductFunctionInterface { + return {}; + } + + searchEngine(): CustomSearchEngine { + return { + workspaceType: { operation: '$eq' }, + workspaceTypes: { dbFName: 'workspaceType', operation: '$in' }, + + equipmentIds: { operation: '$in' }, + featureIds: { operation: '$in' }, + + pricingPerHour: { + operation: 'between', + beetweenConfig: { + inputFromField: 'priceMin', + inputToField: 'priceMax', + }, + }, + + from : { + operation:'$gte', + dbFName:'availability.dates.endDate' + }, + + date: { + operation: 'dateRange', + dateRangeConfig: { + dbFromField: 'availability.dates.startDate', + dbToField: 'availability.dates.endDate', + inputFromField: 'dateFrom', + inputToField: 'dateTo', + dateRangeType: RangeType.included, + }, + }, + + hour: { + operation: 'hourRange', + hourRangeConfig: { + dbMainField: 'availability.hours', + dbSubfieldFrom: 'from', + dbSubfieldTo: 'to', + inputFromField: 'hoursFrom', + inputToField: 'hoursTo', + dateRangeType: RangeType.included, + }, + }, + + maxCapacity: { + operation: '$gte', + }, + + createdBy: { + operation: '$eq', + }, + }; + } +} + +@ArgsType() +export class WorkspaceArgs extends BaseGeoArgs { + @Field({ nullable: true }) + dateFrom?: Date; + @Field({ nullable: true }) + dateTo?: Date; + + @Field({ nullable: true }) + hoursFrom?: string; + @Field({ nullable: true }) + hoursTo?: string; + + @Field(() => WorkspaceTypeEnum, { nullable: true }) + workspaceType?: WorkspaceTypeEnum; + @Field(() => [WorkspaceTypeEnum], { nullable: true }) + workspaceTypes?: WorkspaceTypeEnum[]; + + @Field(() => Int, { nullable: true }) + priceMin?: number; + @Field(() => Int, { nullable: true }) + priceMax?: number; + @Field(() => Int, { nullable: true }) + maxCapacity?: number; + + @Field(() => [String], { nullable: true }) + equipmentIds?: string[]; + @Field(() => [String], { nullable: true }) + featureIds?: string[]; +} + +@InputType() +export class NewWorkspaceInput extends NewBaseSEOSimpleInput { + @Field(() => WorkspaceTypeEnum) + workspaceType: WorkspaceTypeEnum; + + @Field(() => Int) + pricingPerHour: number; + @Field(() => AvailableCurrency) + currency: AvailableCurrency; + + @Field(() => Int) + maxCapacity: number; + + @Field(() => Int) + minStay: number; + @Field(() => Int, { nullable: true }) + maxStay?: number; + + @Field(() => AvailabilityComponent) + availability: AvailabilityComponent; + + @Field(() => MainImageComponent) + mainImages: MainImageComponent; + + @Field(() => [String]) + equipmentIds: string[]; + @Field(() => [String]) + featureIds: string[]; + + @Field(() => AddressStrictComponent) + address:AddressStrictComponent + + place: { loc: any; formattedAddress: string; }; +} + +@InputType() +export class EditWorkspaceInput extends EditBaseSEOSimpleInput { + @Field(() => WorkspaceTypeEnum, { nullable: true }) + workspaceType?: WorkspaceTypeEnum; + + @Field(() => Int, { nullable: true }) + pricingPerHour?: number; + @Field(() => AvailableCurrency) + currency: AvailableCurrency; + + @Field(() => Int, { nullable: true }) + maxCapacity?: number; + + @Field(() => Int, { nullable: true }) + minStay?: number; + @Field(() => Int, { nullable: true }) + maxStay?: number; + + @Field(() => AvailabilityComponent, { nullable: true }) + availability?: AvailabilityComponent; + + @Field(() => MainImageComponent) + mainImages: MainImageComponent; + + @Field(() => [String], { nullable: true }) + equipmentIds?: string[]; + @Field(() => [String], { nullable: true }) + featureIds?: string[]; + + @Field(() => AddressStrictComponent, {nullable:true}) + address:AddressStrictComponent + place: { loc: any; formattedAddress: string; }; +} diff --git a/src/workspace/workspace.resolver.admin.ts b/src/workspace/workspace.resolver.admin.ts new file mode 100644 index 0000000..f4c3d07 --- /dev/null +++ b/src/workspace/workspace.resolver.admin.ts @@ -0,0 +1,29 @@ +// import { Resolver, Query, FieldResolver, Root, Arg, Mutation, Ctx, Args, Authorized, ResolverInterface } from 'type-graphql'; + +// import { createBaseResolver } from '@seed/graphql/baseResolvers/BaseResolver'; +// import { AccountTypeEnum } from '@src/accounts/account.components'; +// import ProductModel from './workspace.model'; + +// const ProductBaseAdminResolver = createBaseResolver('products', ProductModel, ProductArgs, NewProductInput, EditProductInput, [ +// AccountTypeEnum.admin, +// ]); + +// @Resolver(ProductModel) +// export default class ProductAdminResolver extends ProductBaseAdminResolver { +// /* +// ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ +// ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ +// ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ +// ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ +// ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ +// ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ +// */ +// /* +// ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ +// ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ +// ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ +// ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ +// ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ +// ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ +// */ +// } diff --git a/src/workspace/workspace.resolver.fields.ts b/src/workspace/workspace.resolver.fields.ts new file mode 100644 index 0000000..0e883f3 --- /dev/null +++ b/src/workspace/workspace.resolver.fields.ts @@ -0,0 +1,62 @@ +import { Resolver, FieldResolver, Root, Ctx, Args } from 'type-graphql'; + +import { ApolloContext } from '@seed/interfaces/context'; +import { oneToManyComplexity } from '@seed/graphql/Middleware'; +import ProductModel from './workspace.model'; +import ListModel from '@services/module-cms/functions/lists/list.model'; +import { BookingEngineSchema } from '@services/module-booking/functions/bookings/schemas/bookings.engine.schema'; +import WorkspaceModel from './workspace.model'; +import { BookingService } from '@services/module-booking/functions/bookings/booking.service'; +import { BookingArgsSchema } from '@src/bookings/schemas/bookings.input'; +import { BookingSchema } from '@src/bookings/schemas/bookings.schema'; +import { WorkspaceOwnerComponent } from '@src/__components/components'; + +@Resolver(WorkspaceModel) +export default class WorkspaceFieldResolver { + /* + ███████╗██╗███████╗██╗ ██████╗ ███████╗ + ██╔════╝██║██╔════╝██║ ██╔══██╗██╔════╝ + █████╗ ██║█████╗ ██║ ██║ ██║███████╗ + ██╔══╝ ██║██╔══╝ ██║ ██║ ██║╚════██║ + ██║ ██║███████╗███████╗██████╔╝███████║ + ╚═╝ ╚═╝╚══════╝╚══════╝╚═════╝ ╚══════╝ + */ + + @FieldResolver(() => [ListModel], { complexity: oneToManyComplexity, nullable: 'itemsAndList' }) + async getEquipments(@Root() product: ProductModel, @Ctx() ctx: ApolloContext): Promise<(ListModel | Error)[]> { + if (product.equipmentIds) return await ctx.ctx.loaders.listLoader.loadMany(product.equipmentIds); + return []; + } + + @FieldResolver(() => [ListModel], { complexity: oneToManyComplexity, nullable: 'itemsAndList' }) + async getFeatures(@Root() product: ProductModel, @Ctx() ctx: ApolloContext): Promise<(ListModel | Error)[]> { + if (product.featureIds) return await ctx.ctx.loaders.listLoader.loadMany(product.featureIds); + return []; + } + + @FieldResolver(() => [BookingSchema], { complexity: oneToManyComplexity, nullable: 'itemsAndList' }) + async getBookings(@Root() workspace: WorkspaceModel, @Args() args: BookingArgsSchema, @Ctx() ctx: ApolloContext): Promise { + return new BookingService().getBookings(args, ctx); + } + + @FieldResolver(() => WorkspaceOwnerComponent, { complexity: oneToManyComplexity, nullable: true }) + async getOwner(@Root() workspace: WorkspaceModel, @Ctx() ctx: ApolloContext): Promise { + if (workspace.createdBy) return await ctx.ctx.loaders.accountLoader.load(workspace.createdBy); + return; + } + + // @FieldResolver(() => [CommentModel], { complexity: oneToManyComplexity }) + // async comments(@Root() product: ProductModel, @Args() args: CommentArgs, @Ctx() ctx: ApolloContext): Promise<(CommentModel | Error)[]> { + // return (await ctx.ctx.loaders.commentRessourceLoader(args.pagination).load(product._id)) as any; + // } + + // // Todo: Optimize + // @FieldResolver(() => [ProductModel], { complexity: oneToManyComplexityNoLoader }) + // async relatedProducts(@Root() product: ProductModel, @Ctx() ctx: ApolloContext): Promise<(ProductModel | Error)[]> { + // return await product.getMany( + // { _id: { $ne: product._id }, categoryIds: { $in: product.categoryIds }, published: true }, + // { limit: 25, skip: 0 }, + // ctx, + // ); + // } +} diff --git a/src/workspace/workspace.resolver.ts b/src/workspace/workspace.resolver.ts new file mode 100644 index 0000000..817d2b0 --- /dev/null +++ b/src/workspace/workspace.resolver.ts @@ -0,0 +1,102 @@ +import { Arg, Args, Authorized, Ctx, Query, Resolver } from 'type-graphql'; + +import { createBaseResolver } from '@seed/graphql/baseResolvers/BaseResolver'; +import { AccountTypeEnum } from '@src/accounts/account.components'; +import WorkspaceModel, { WorkspaceArgs, EditWorkspaceInput, NewWorkspaceInput } from './workspace.model'; +import { getCountGeneric, getManyGenericWithArgs, getOneGeneric } from '@seed/graphql/BaseService'; +import { ApolloContext } from '@seed/interfaces/context'; +import { auth } from 'firebase-admin'; +import { domain } from 'process'; +import { BookingService } from '@services/module-booking/functions/bookings/booking.service'; +import { DateTime } from 'luxon'; +import { converDateRangeToDateTime, isTimeSlotInDates } from '@seed/helpers/Utils.dates'; +import { BookingsRessourceEnum } from '@src/__components/components'; +import moment from 'moment'; + +const WorkspaceBaseResolver = createBaseResolver('workspaces', WorkspaceModel, WorkspaceArgs, NewWorkspaceInput, EditWorkspaceInput, [ + AccountTypeEnum.user, +]); +@Resolver(WorkspaceModel) +export default class WorkspaceResolver extends WorkspaceBaseResolver { + /* + ██████╗ ██╗ ██╗███████╗██████╗ ██╗ ██╗ + ██╔═══██╗██║ ██║██╔════╝██╔══██╗╚██╗ ██╔╝ + ██║ ██║██║ ██║█████╗ ██████╔╝ ╚████╔╝ + ██║▄▄ ██║██║ ██║██╔══╝ ██╔══██╗ ╚██╔╝ + ╚██████╔╝╚██████╔╝███████╗██║ ██║ ██║ + ╚══▀▀═╝ ╚═════╝ ╚══════╝╚═╝ ╚═╝ ╚═╝ + */ + + @Query(() => WorkspaceModel) + async workspacesGetOne(@Arg('id') id: string, @Ctx() ctx: ApolloContext): Promise { + const model = new WorkspaceModel(); + return getOneGeneric(model, id, ctx); + } + + @Query(() => [WorkspaceModel]) + async workspacesSearchMany(@Args() args: WorkspaceArgs, @Ctx() ctx: ApolloContext): Promise { + const model = new WorkspaceModel(); + + // (1) Get the bookings in those dates + if (args.dateFrom && args.dateTo) { + const workspaceIdToOmit: any[] = await new BookingService().getBookingsToFilter(BookingsRessourceEnum.workspaces, { + startDate: args.dateFrom, + endDate: args.dateTo, + startTime: args.hoursFrom || '00:00', + endTime: args.hoursTo || '23:59', + }); + if (workspaceIdToOmit.length > 0) (args as any).n_ids = workspaceIdToOmit; + } + + // At least availability in the future + if (!args.dateFrom) (args as any).from = moment().toDate(); + + // (end) Get the workspaces without those above + const workspaces = await getManyGenericWithArgs(model, args, ctx); + return workspaces; + } + + @Query(() => Number) + async workspacesGetCount(@Args() args: WorkspaceArgs, @Ctx() ctx: ApolloContext): Promise { + const model = new WorkspaceModel(); + + // (1) Get the bookings in those dates + if (args.dateFrom && args.dateTo) { + const workspaceIdToOmit: any[] = await new BookingService().getBookingsToFilter(BookingsRessourceEnum.workspaces, { + startDate: args.dateFrom, + endDate: args.dateTo, + startTime: args.hoursFrom || '00:00', + endTime: args.hoursTo || '23:59', + }); + if (workspaceIdToOmit.length > 0) (args as any).n_ids = workspaceIdToOmit; + } + + // At least availability in the future + if (!args.dateFrom) (args as any).from = moment().toDate(); + + // (end) Get the workspaces without those above + return getCountGeneric(model, args, ctx); + } + + @Query(() => [WorkspaceModel]) + @Authorized(AccountTypeEnum.user) + async workspacesGetMine(@Args() args: WorkspaceArgs, @Ctx() ctx: ApolloContext): Promise { + const model = new WorkspaceModel(); + return getManyGenericWithArgs(model, { ...args, createdBy: ctx.ctx.user._id }, ctx); + } + @Query(() => Number) + @Authorized(AccountTypeEnum.user) + async workspacesGetMineCount(@Args() args: WorkspaceArgs, @Ctx() ctx: ApolloContext): Promise { + const model = new WorkspaceModel(); + return getCountGeneric(model, { ...args, createdBy: ctx.ctx.user._id }, ctx); + } + + /* + ███╗ ███╗██╗ ██╗████████╗ █████╗ ████████╗ ██████╗ ██████╗ ███████╗ + ████╗ ████║██║ ██║╚══██╔══╝██╔══██╗╚══██╔══╝██╔═══██╗██╔══██╗██╔════╝ + ██╔████╔██║██║ ██║ ██║ ███████║ ██║ ██║ ██║██████╔╝███████╗ + ██║╚██╔╝██║██║ ██║ ██║ ██╔══██║ ██║ ██║ ██║██╔══██╗╚════██║ + ██║ ╚═╝ ██║╚██████╔╝ ██║ ██║ ██║ ██║ ╚██████╔╝██║ ██║███████║ + ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ + */ +} diff --git a/tsconfig.json b/tsconfig.json new file mode 100644 index 0000000..c09e529 --- /dev/null +++ b/tsconfig.json @@ -0,0 +1,34 @@ +{ + "compilerOptions": { + "preserveConstEnums": true, + "strictNullChecks": true, + "inlineSources": true, + "inlineSourceMap": true, + "sourceRoot": "/", + "target": "es2017", + "outDir": ".build", + "module": "commonjs", + "moduleResolution": "node", + "esModuleInterop": true, + "lib": ["es2020", "DOM", "DOM.Iterable"], + "rootDir": "./", + "baseUrl": "./", + "resolveJsonModule": true, + "emitDecoratorMetadata": true, + "experimentalDecorators": true, + "skipLibCheck": true, + "paths": { + "@config/*": ["config/*"], + "@seed/*": ["lib/seed/*"], + "@src/*": ["src/*"], + "@services/*": ["services/*"], + "@lib/*": ["lib/*"], + + } + }, + "exclude": [ + "**/__tests/*", + "*.spec.ts", + "**/examples/*", + ] +}